反射中,Class.forName和ClassLoader区别

今天浏览其他博客的时候,看到一个问题问“反射中,Class.forName和ClassLoader区别”,思考后,发现自己知道这两个都可以用来进行类的加载,但是其区别确实还不了解,经查阅相关资料学习后总结如下:

一、类加载过程

装载:通过累的全限定名获取二进制字节流,将二进制字节流转换成方法区中的运行时数据结构,在内存中生成Java.lang.class对象; 

链接:执行下面的校验、准备和解析步骤,其中解析步骤是可以选择的; 

  校验:检查导入类或接口的二进制数据的正确性;(文件格式验证,元数据验证,字节码验证,符号引用验证) 

  准备:给类的静态变量分配并初始化存储空间; 

  解析:将常量池中的符号引用转成直接引用; 

初始化:激活类的静态变量的初始化Java代码和静态Java代码块,并初始化程序员设置的变量值。

二、两种加载类方式的区别

Java中Class.forName和classloader都可以用来对类进行加载。

  •   Class.forName(“className”);

        其实这种方法调运的是:Class.forName(className, true, ClassLoader.getCallerClassLoader())方法
        参数一:className,需要加载的类的名称。
        参数二:true,是否对class进行初始化(需要initialize)
        参数三:classLoader,对应的类加载器

  •    ClassLoader.laodClass(“className”);

        其实这种方法调运的是:ClassLoader.loadClass(name, false)方法
        参数一:name,需要加载的类的名称
        参数二:false,这个类加载以后是否需要去连接(不需要linking)

可见Class.forName除了将类的.class文件加载到jvm中之外,还会对类进行解释,执行类中的static块。

而classloader只干一件事情,就是将.class文件加载到jvm中,不会执行static中的内容,只有在newInstance才会去执行static块。

所以这就是为什么在jdbc进行数据库操作的时候要用Class.forName进行驱动的加载。

Class.forName(“com.mysql.jdbc.Driver”);//通过这种方式将驱动注册到驱动管理器上
Connection conn = DriverManager.getConnection(“url”,“userName”,“password”);//通过驱动管理器获得相应的连接
.............
.............

再来看一下com.mysql.jdbc.Driver的源码,其中有一段静态代码块:

public class Driver extends NonRegisteringDriver
  implements java.sql.Driver
{
    //注意,这里有一个static的代码块,这个代码块将会在class初始化的时候执行
  static
  {
    try
    {
        //将这个驱动Driver注册到驱动管理器上
      DriverManager.registerDriver(new Driver());
    } catch (SQLException E) {
      throw new RuntimeException(“Can’t register driver!”);
    }
  }
}

Class.forName(“com.mysql.jdbc.Driver”)会进行class的初始化,执行static代码块。
也就是说class初始化以后,就会将驱注册到DriverManageer上,之后才能通过DriverManager去获取相应的连接。
但是要是我们使用ClassLoader.loadClass(com.mysql.jdbc.Driver)的话,不会link,更也不会初始化class。
相应的就不会回将Driver注册到DriverManager上面,后面肯定不能通过DriverManager获取相应的连接。

你可能感兴趣的:(java,反射,Class.forName,classLoader)