ClassLoader.loadClass和Class.forName的区别

为何要把ClassLoader.loadClass(String name)和Class.forName(String name)进行比较呢,由于他们都能在运行时对任意一个类,都可以知道该类的全部属性和方法;对于任意一个对象,都可以调用它的任意方法和属性。java

在比较它俩以前需先了解一下java类装载的过程mysql

java类装载过程分为3步:sql

   

 

  1:加载数据库

    Jvm把class文件字节码加载到内存中,并将这些静态数据装换成运行时数据区中方法区的类型数据,在运行时数据区堆中生成一个表明这个类安全

  的java.lang.Class对象,做为方法区类数据的访问入口。jvm

  *释:方法区不单单是存放方法,它存放的是类的类型信息。spa

  2:连接:执行下面的校验、准备和解析步骤,其中解析步骤是可选的对象

    a:校验:检查加载的class文件的正确性和安全性blog

    b:准备:为类变量分配存储空间并设置类变量初始值,类变量随类型信息存放在方法区中,生命周期很长,使用不当和容易形成内存泄漏。生命周期

    *:类变量就是static变量;初始值指的是类变量类型的默认值而不是实际要赋的值

    c:解析:jvm将常量池内的符号引用转换为直接引用

  3:初始化:执行类变量赋值和静态代码块

 

在了解了类装载过程以后咱们继续比较两者区别:

  • Classloder.loaderClass(String name)

    其实该方法内部调用的是:Classloder. loadClass(name, false)

    方法:Classloder. loadClass(String name, boolean resolve)

        1:参数name表明类的全限定类名

        2:参数resolve表明是否解析,resolve为true是解析该类

  • Class.forName(String name)

    其实该方法内部调用的是:Class.forName(className, true, ClassLoader.getClassLoader(caller))

    方法:Class.forName0(String name, boolean initialize, ClassLoader loader)

      参数name表明全限定类名

      参数initialize表示是否初始化该类,为true是初始化该类

      参数loader 对应的类加载器

  • 二者最大的区别

    Class.forName获得的class是已经初始化完成的

    Classloder.loaderClass获得的class是尚未连接的

  • 怎么使用

    有些状况是只须要知道这个类的存在而不须要初始化的状况使用Classloder.loaderClass,而有些时候又必须执行初始化就选择Class.forName

    例如:数据库驱动加载就是使用Class.froName(“com.mysql.jdbc.Driver”),

    下面咱们来看看Driver的源代码:

public class Driver extends NonRegisteringDriver implements java.sql.Driver {

    public Driver() throws SQLException {

    } 
static { try { DriverManager.registerDriver(new Driver()); } catch (SQLException var1) { throw new RuntimeException("Can\'t register driver!"); } } }

    从Driver的源码中咱们能够看出Driver这个类只有一个static块,这样咱们须要初始化后才能获得DriverManager,因此咱们选择使用Class.forName()

相关文章
相关标签/搜索