分析Class.forName()和ClassLoader.loadClass在数据库连接的应用

4 篇文章 0 订阅

一 Java类装载过程

1.装载:通过累的全限定名获取二进制字节流,将二进制字节流转换成方法区中的运行时数据结构,在内存中生成Java.lang.class对象; 
2.链接:执行下面的校验、准备和解析步骤,其中解析步骤是可以选择的; 
  校验:检查导入类或接口的二进制数据的正确性;(文件格式验证,元数据验证,字节码验证,符号引用验证) 
  准备:给类的静态变量分配并初始化存储空间; 
  解析:将常量池中的符号引用转成直接引用; 
3.初始化:激活类的静态变量的初始化Java代码和静态Java代码块,并初始化程序员设置的变量值。

二 分析 Class.forName()和ClassLoader.loadClass

Class.forName(className)方法,内部实际调用的方法是  Class.forName(className,true,classloader);第2个boolean参数表示类是否需要初始化,  Class.forName(className)默认是需要初始化。一旦初始化,就会触发目标对象的 static块代码执行,static参数也也会被再次初始化。

ClassLoader.loadClass(className)方法,内部实际调用的方法是  ClassLoader.loadClass(className,false);第2个 boolean参数,表示目标对象是否进行链接,false表示不进行链接,由上面介绍可以,不进行链接意味着不进行包括初始化等一些列步骤,那么静态块和静态对象就不会得到执行。

三  数据库链接为什么使用Class.forName(className)

 JDBC连接数据库的步骤为:

// 首先要导入数据库驱动jar包、直接复制到文件那里,然后加入到路径  
// 1. 注册驱动  
Class.forName("com.mysql.jdbc.Driver");java反射机制  
//2. 创建一个连接对象  
Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/testdb","root","password");  
//3. 创建一个sql语句的发送命令对象  
Statement stmt = conn.createStatement();  
//4. 执行sql,拿到查询的结果集对象  
ResultSet rs = stmt.executeQuery("select * from user");  
//5. 输出结果集的数据  
while(rs.next()){  
    system.out.println(rs.getInt("id")+":"+rs.getString("name"));  
}  
//6. 关闭连接,命令对象以及结果集。  
rs.close();  
stmt.close();  
conn.close(); 

在学习JDBC的时候,通常有两种方式去注册数据库驱动程序(这里采用MySQL数据库),分别为:

DriverManager.registerDriver(new Driver());  
Class.forName("com.mysql.jdbc.Driver");  

 那么这两种方式有什么异同呢?首先我们到DriverManager中去看一下,

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的源代码。

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的方式我们可以在运行的时候再动态改变。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值