当前位置:Gxlcms > 数据库问题 > 关于Java的jdbc中 DriverManager.registerDriver(driver); //注册驱动 有没有必要写的思考

关于Java的jdbc中 DriverManager.registerDriver(driver); //注册驱动 有没有必要写的思考

时间:2021-07-01 10:21:17 帮助过:9人阅读

加载数据库驱动的时候,有如下部分代码:

/1) 注册驱动程序
//给java.sql.Driver接口的引用赋值 com.mysql.jdbc.Driver 实现类对象
// Driver driver = new com.mysql.jdbc.Driver();
// DriverManager.registerDriver(driver); //注册驱动
先到数据库驱动包看下这个Driver类:

技术图片可以看到这个类里面有个static语句块,这里先来复习以下,什么时候会加载static语句块,有如下情况:

 

 

当一个类被主动使用时,Java虚拟就会对其初始化,如下六种情况为主动使用:

  1. 当创建某个类的新实例时(如通过new或者反射,克隆,反序列化等)

  2. 当调用某个类的静态方法时

  3. 当使用某个类或接口的静态字段时

  4. 当调用Java API中的某些反射方法时,比如类Class中的方法,或者java.lang.reflect中的类的方法时

  5. 当初始化某个子类时

  6. 当虚拟机启动某个被标明为启动类的类(即包含main方法的那个类)

    Java编译器会收集所有的类变量初始化语句和类型的静态初始化器,将这些放到一个特殊的方法中:clinit。

 特别要注意的是:使用final修饰的静态字段,在调用的时候不会对类进行初始化!以及类被加载了不一定就会执行静态代码块,只有一个类被主动使用的时候,静态代码才会被执行

那么回到原来的问题,在创建Driver这个对象的时候:

Driver driver = new com.mysql.jdbc.Driver();

必然会执行其内的static语句块的,其static语句块的作用就和:

DriverManager.registerDriver(driver); 

这段代码的作用是一样的,所以有必要再写一次这行代码吗?

我试着删除了这行代码,发现还是能顺利连接到数据库。

我写了一段代码计算一下driver的数量,如下所示:

public class jbdctest01 {
public static void main(String[] args) throws ClassNotFoundException, SQLException {
//1) 注册驱动程序
//给java.sql.Driver接口的引用赋值 com.mysql.jdbc.Driver 实现类对象
// Driver driver = new com.mysql.jdbc.Driver();
// DriverManager.registerDriver(driver); //注册驱动
//注册驱动也可以使用Class.forName()创建Class对象时,会执行类的静态代码块, 在com.mysql.jdbc.Driver类的静态代码块中,注册驱动
Class.forName("com.mysql.jdbc.Driver");
//2)在当前程序与 数据库服务器建立连接
// Connection connection = DriverManager.getConnection("jdbc:mysql://数据库服务器IP地址:MySQL服务的端口号/数据库名", "登录服务器的用户名", "登录密码");
Enumeration<Driver> drivers = DriverManager.getDrivers();

int nums = 0;
//使用while循环来遍历驱动的个数
while(drivers.hasMoreElements()) {
nums ++;
//打印出驱动
System.out.println(drivers.nextElement());
}
//打印出驱动个数
System.out.println("驱动个数:" + nums);
}
}
测试结果如图所示:

技术图片

 

用反射的方法加载驱动,驱动数为一个,这是意料之中的事情,在创建Driver这个对象时,会执行static中的代码块,因此会加载驱动,且数量为1

接下来测试,另外一种方法,代码如下:

public class jbdctest01 {
    public static void main(String[] args) throws ClassNotFoundException, SQLException {
        //1) 注册驱动程序
        //给java.sql.Driver接口的引用赋值 com.mysql.jdbc.Driver 实现类对象
            Driver driver = new com.mysql.jdbc.Driver();

            DriverManager.registerDriver(driver);   //注册驱动
        //注册驱动也可以使用Class.forName()创建Class对象时,会执行类的静态代码块, 在com.mysql.jdbc.Driver类的静态代码块中,注册驱动
       //Class.forName("com.mysql.jdbc.Driver");
        //2)在当前程序与 数据库服务器建立连接
//        Connection connection = DriverManager.getConnection("jdbc:mysql://数据库服务器IP地址:MySQL服务的端口号/数据库名", "登录服务器的用户名", "登录密码");
        Enumeration<Driver> drivers = DriverManager.getDrivers();

        int nums = 0;
        //使用while循环来遍历驱动的个数
        while(drivers.hasMoreElements()) {
            nums ++;
            //打印出驱动
            System.out.println(drivers.nextElement());
        }
        //打印出驱动个数
        System.out.println("驱动个数:" + nums);
  }
}
测试结果如图所示:

 

 技术图片

 

可以看到结果是2

接下来注释掉:DriverManager.registerDriver(driver); 这行代码,可以看到输出结果:

技术图片

 

可以看到驱动个数为1个,说明了DriverManager.registerDriver(driver);这行代码可以去掉,如果加上,就会出现了两个驱动,因为在创建Driver这个对象的时候,其内的static语句块会自动执行了。

至此得出结论了。

另外还有一个小的疑问,就是一个对象执行的时候,是先创建类还是先执行其内的静态方法呢?我特意写了一个程序试了一下:

public class Teststatic {
public Teststatic(){
System.out.println("看看谁先被执行");

}
static {
System.out.println("我被执行了,哈哈哈");
}
}
class T {
public static void main(String[] args) {
Teststatic a = new Teststatic();

}
}
运行结果如下:

技术图片

 

说明static语句块比构造方法执行的要早,我记得也是这样的。那么,我就有了一个疑问,因为Driver中的static,要用到Driver的实现类,如果先执行static语句块的话,不会抛出异常吗?因为这个时候Driver的实现类还没创建好啊

这里我犯了一个错误,就是误认为构造方法执行完毕之后类才被创建,其实不是的,是类创建好之后才会执行构造方法,所以是不会报错的。

 



 

关于Java的jdbc中 DriverManager.registerDriver(driver); //注册驱动 有没有必要写的思考

标签:就会   except   数据库   反序   mysq   tst   gis   问题   alt   

人气教程排行