在学习JDBC的时候,通常有两种方式去注册数据库驱动程序(这里采用MySQL数据库),分别为:
Java代码
DriverManager.registerDriver(new Driver());
Class.forName("com.mysql.jdbc.Driver");
那么这两种方式有什么异同呢?首先我们到DriverManager中去看一下,
Java代码
public static synchronized void registerDriver(java.sql.Driver driver)
throws SQLException {
if (!initialized) {
initialize();
}
DriverInfo di = new DriverInfo();
di.driver = driver;
di.driverClass = driver.getClass();
di.driverClassName = di.driverClass.getName();
// Not Required -- drivers.addElement(di);
writeDrivers.addElement(di);
println("registerDriver: " + di);
/* update the read copy of drivers vector */
readDrivers = (java.util.Vector) writeDrivers.clone();
}
很明显,DriverManager将我们需要注册的驱动程序信息封装成DriverInfo放进了一个writeDrivers中,这个writeDrivers是DriverManager中声明的一个static类型Vector变量。在getConnection的时候会再用到。
那么Class.forName("com.mysql.jdbc.Driver")是如何注册驱动程序的呢,我们知道Class.forName("类名")的主要作用是向虚拟机实例化一个Class实例,我们看一下com.mysql.jdbc.Driver的源代码。
Java代码
public class Driver extends NonRegisteringDriver implements java.sql.Driver {
// ~ Static fields/initializers
// ---------------------------------------------
//
// Register ourselves with the DriverManager
//
static {
try {
java.sql.DriverManager.registerDriver(new Driver());
} catch (SQLException E) {
throw new RuntimeException("Can't register driver!");
}
}
在 com.mysql.jdbc.Driver中有一段静态代码块,是向 DriverManager注册一个Driver实例。这样在 Class.forName("com.mysql.jdbc.Driver")的时候,就会首先去执行这个静态代码块,于是和DriverManager.registerDriver(new Driver())有了相同的效果。
那么对于这两种方法,在这里,我推荐使用第二种,即Class.forName("类名")的方式。原因有两点
1、在我们执行DriverManager.registerDriver(new Driver())的时候,静态代码块也已经执行了,相当于是实例化了两个Driver对象。
2、 DriverManager.registerDriver(new Driver())产生了一种对MySQL的一种依赖。而Class.forName的方式我们完全可以在运行的时候再动态改变。
所有JDBC的书都跟我们说使用Class.forName可以强制装载jdbc Driver,但从来没有人告诉我们是如何实现的。这一说法最早应该由think in java给出,而这一点一直让我疑惑。
Class.forName用于将class装载入内存,建立一个相应的class类实例,然后返回句柄。如果不用这个方法来明确建立class类实例,而直接用new来建立实际的类实例,则在建立实例前仍然会用classloader来得到class实例(如果内存中没有),然后再建立类实例,因此用getClass方法可以得到class实例。
我们知道在建立实例时,会调用其构建函数,这称为初始化,这一步骤包括对其属性的初始化。但是如果不建立实例,而建立相应的class类实例,就象Class.forName所作的那样,又会做哪些工作呢?对于那些静态变量(属性)的初始化将在该阶段完成,这也就是为什么在不建立实例的时候就可以调用静态变量的原因。另外一些静态代码,即static{}中的代码也将在这里执行。
最初我认为自动注册的代码由jvm完成,它将检测装载的类,如果实现driver接口,就直接调用DriverManager.registerDriver(new Driver())来完成注册,但是这显然与classloader本身需要完成的工作无关,那么由DriverManager来完成检测吗?这就需要一个专门用于检测的线程,显然会影响jvm的性能。因此另一个实现方案就是使用静态代码,则在driver的实现类中必然有如下代码static{DriverManager.registerDriver(new Driver());},该代码在装载class文件建立class类时自动执行完成注册。
那么事实究竟如何呢?根据我们的推测,如果使用DriverManager.registerDriver(new Driver())来显式注册,则实际上会注册两次,一次是在装载class类时由代码static{DriverManager.registerDriver(new Driver());}来完成,一次则是执行我们写的DriverManager.registerDriver(new Driver())代码。
这是用DriverManager.registerDriver(new JdbcOdbcDriver())来显式注册时,用DriverManager.getDrivers()得到的注册的drivers,
sun.jdbc.odbc.JdbcOdbcDriver@42e816
sun.jdbc.odbc.JdbcOdbcDriver@9304b1
显然sun.jdbc.odbc.JdbcOdbcDriver被注册了两次。
因此在注册driver时,用Class.forName是最好的方法,它保证了DriverManager中的driver的唯一性。DriverManager.registerDriver(new JdbcOdbcDriver())虽然不会产生代码错误,但显然会影响执行效率,因为在依次查询driver时,多个相同的driver的存在显然会导致查询量的增加。
分享到:
相关推荐
Java中的`Class.forName()`方法是一个非常关键的工具,主要用于动态加载类。它的作用在于根据给定的类全名(包括包名)返回对应的`Class`对象。`Class`对象是Java反射机制的基础,它代表着运行时的类信息,可以通过...
在Java编程中,`Class.forName()`, `newInstance()` 和 `new` 关键字都是与对象创建和类加载紧密相关的概念,但它们各自有着不同的用途和限制。 首先,`Class.forName("className")` 是一个静态方法,它用于在运行...
总的来说,`Class.forName()` 在JDBC中的作用是加载并初始化数据库驱动,使得`DriverManager` 可以识别并管理这个驱动,进而允许我们创建数据库连接。这是一种简洁且有效的做法,也是JDBC编程的标准实践。
`Class.forName()`方法内部调用了本地方法`forName0`,并传入了一个布尔参数`initialize`,默认值为`true`,表示在加载类之后执行初始化。初始化阶段不仅包括对类变量的赋值,还包括执行静态初始化块,对于父类也是...
- **Class.forName()**:用于加载特定的JDBC驱动类。 - **DriverManager.getConnection()**:用于建立到数据库的连接。 - **URL**:指定数据库的位置以及连接参数。 - **Connection**:表示与数据库之间的连接。 ##...
Class.forName("oracle.jdbc.driver.OracleDriver").newInstance(); String url = "jdbc:oracle:thin:@localhost:1521:orcl"; // orcl 为数据库SID String user = "test"; String password = "test"; ...
如果熟悉使用JDBC来连接数据库的同学一定很清楚连接数据库的代码中一定会有依据Class.forName (com.mysql.jdbc.Driver); public static Connection getConnection() throws ClassNotFoundException, SQLException ...
Class.forName("oracle.jdbc.driver.OracleDriver").newInstance(); //Sql Server7.0/2000 数据库 Class.forName("com.microsoft.jdbc.sqlserver.SQLServerDriver").newInstance(); //DB2 数据库 Class.froName(...
1. **加载数据库驱动**:通过`Class.forName()`方法加载特定数据库的JDBC驱动。 2. **建立连接**:使用`DriverManager.getConnection()`方法获取数据库连接。 3. **创建Statement对象**:通过连接对象创建Statement...
在Java编程中,有时我们需要与Microsoft Access数据库进行交互,特别是在处理小型数据存储或与遗留系统集成时。然而,Java的标准数据库连接API(JDBC)原生并不支持Access数据库。为了解决这个问题,开发者通常会...
Class.forName(driver).newInstance(); Connection con = DriverManager.getConnection(url); Statement stm = con.createStatement(); String sql = "select * from sheet"; for(int i=0;i;i++){ ResultSet ...
这是MySQL 5.7版本以上数据库的连接驱动jar包,对应加载驱动程序代码: Class.forName("com.mysql.cj.jdbc.Driver"); 该驱动包适用于Java web开发,连接MySQL数据库时导入工程。 参考代码: Class.forName(...
这是MySQL 5.7版本以上数据库的连接驱动jar包,对应加载驱动程序代码: Class.forName("com.mysql.cj.jdbc.Driver"); 该驱动包适用于Java web开发,连接MySQL数据库时导入工程。 参考代码: Class.forName(...
这是MySQL 5.7版本以上数据库的连接驱动jar包(解压密码1234),对应加载驱动程序代码: Class.forName("com.mysql.cj.jdbc.Driver"); 该驱动包适用于Java web开发,连接MySQL数据库时导入工程。 参考代码: Class....
Class.forName("sun.jdbc.odbc.JdbcOdbcDriver") String driver = "jdbc:odbc:try" Connection conn = DriverManager.getConnection(driver,"sa","") //如果是windows验证,则不用用户名和密码
当`Class.forName()`执行后,驱动程序会被注册到`DriverManager`中,使得之后可以通过`DriverManager.getConnection()`方法建立数据库连接。 其次,`DriverManager.registerDriver()`方法允许我们手动注册一个`java...
通常,我们使用`Class.forName()`方法来加载驱动,例如: ```java Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver"); ``` 如果这个方法抛出ClassNotFoundException,表示驱动类没有找到。 4. **...