• ClassLoader.loadClass和Class.forName的区别


    为什么要把ClassLoader.loadClass(String name)和Class.forName(String name)进行比较呢,因为他们都能在运行时对任意一个类,都能够知道该类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性。

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

    java类装载过程分为3步:

       

      1:加载

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

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

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

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

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

        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 对应的类加载器

    • 两种方式的区别
              forName("")得到的class是已经初始化完成的
              loadClass("")得到的class是还没有连接的
              一般情况下,这两个方法效果一样,都能装载Class。
              但如果程序依赖于Class是否被初始化,就必须用Class.forName(name)了。

      • 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表示不进行链接,由上面介绍可以,
        
        不进行链接意味着不进行包括初始化等一些列步骤,那么静态块和静态对象就不会得到执行
        

          

    • 3.举例说明他们各自的使用方法
          java使用JDBC连接数据库时候,我们首先需要加载数据库驱动。
          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获取相应的连接。

  • 相关阅读:
    资源 | TensorFlow推出新工具Seedbank:即刻使用的预训练模型库【转】
    Vim 基本設置 – 使用Vim-plug管理插件 (3)【转】
    Linux kernel 编译问题记录【转】
    深度学习(四)卷积神经网络入门学习(1)【转】
    深度学习:Keras入门(二)之卷积神经网络(CNN)【转】
    深度学习:Keras入门(一)之基础篇【转】
    CNN笔记:通俗理解卷积神经网络【转】
    [Deep Learning] 神经网络基础【转】
    一文弄懂神经网络中的反向传播法——BackPropagation【转】
    Testin
  • 原文地址:https://www.cnblogs.com/1020182600HENG/p/7352735.html
Copyright © 2020-2023  润新知