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获取相应的连接。

时间: 2024-10-14 10:39:30

ClassLoader.loadClass和Class.forName的区别的相关文章

Class.forName和ClassLoader.loadClass的区别(转载)

Class的装载分了三个阶段,loading,linking和initializing,分别定义在The Java Language Specification的12.2,12.3和12.4.Class.forName(className) 实际上是调用Class.forName(className, true, this.getClass().getClassLoader()).注意第二个参数,是指Class被loading后是不是必须被初始化.ClassLoader.loadClass(cl

Class.forName和ClassLoader.loadClass的区别

Class的装载分了三个阶段,loading,linking和initializing,分别定义在The Java Language Specification的12.2,12.3和12.4. Class.forName(className)实际上是调用Class.forName(className, true, this.getClass().getClassLoader()).注意第二个参数,是指Class被loading后是不是必须被初始化. ClassLoader.loadClass(c

Class.forName和ClassLoader.loadClass等

Class类 首先,Class类里可以记载所有类的属性.方法等信息.这个也就是运行时类别标记,它记录了所有的对象(比如int,MyClass,void,数组等等)对应的类信息. Class对象 JVM对于每个所使用的对象,都会有相应的Class对象与之对应,要么这种类型的Class对象已经创建好了,要么就是要新创建一个Class对象. ClassLoader类 ClassLoader类负责的是加载一个类.也就是说,给定这个类相关的名字,定位到或者产生这个类相关的定义信息.ClassLoader对

JVM思考-ClassLoader.loadClasshe和Class.forName区别

JVM思考-ClassLoader.loadClasshe和Class.forName区别 目录:JVM总括:目录 见博客第四节:JVM总括四-类加载过程.双亲委派模型.对象实例化过程 原文地址:https://www.cnblogs.com/java-zzl/p/9905223.html

JAVA 笔记 ClassLoader.getResourceAsStream() 与 Class.getResourceAsStream()的区别

   Class.getResourceAsStream() 会指定要加载的资源路径与当前类所在包的路径一致. 例如你写了一个MyTest类在包com.test.mycode 下,那么MyTest.class.getResourceAsStream("name") 会在com.test.mycode包下查找相应的资源. 如果这个name是以 '/' 开头的,那么就会从classpath的根路径下开始查找. ClassLoader.getResourceAsStream()  无论要查找

【转】 java中Class对象详解和类名.class, class.forName(), getClass()区别

Class对象的生成方式如下: 1.类名.class           说明: JVM将使用类装载器, 将类装入内存(前提是:类还没有装入内存),不做类的初始化工作.返回Class的对象 2.Class.forName("类名字符串")  (注:类名字符串是包名+类名)  说明:装入类,并做类的静态初始化,返回Class的对象 3.实例对象.getClass()  说明:对类进行静态初始化.非静态初始化:返回引用运行时真正所指的对象(因为:子对象的引用可能会赋给父对象的引用变量中)所

java中类名.class, class.forName(), getClass()区别

Class对象的生成方式如下: 1.类名.class           说明: JVM将使用类装载器, 将类装入内存(前提是:类还没有装入内存),不做类的初始化工作.返回Class的对象 2.Class.forName("类名字符串")  (注:类名字符串是包名+类名)  说明:装入类,并做类的静态初始化,返回Class的对象 3.实例对象.getClass()  说明:对类进行静态初始化.非静态初始化:返回引用o运行时真正所指的对象(因为:子对象的引用可能会赋给父对象的引用变量中)

ClassLoader.getResourceAsStream() 与 Class.getResourceAsStream()的区别

package com.xinwei.util; import java.io.IOException; import java.io.InputStream; import java.util.Properties; public class PropertiesLoader { public static void main(String[] args) { // test1(); test2(); } //在使用Class.getResourceAsStream 时, 资源路径有两种方式,

java中Class对象详解和类名.class, class.forName(), getClass()区别

Class对象的生成方式如下: 1.类名.class           说明: JVM将使用类装载器, 将类装入内存(前提是:类还没有装入内存),不做类的初始化工作.返回Class的对象 2.Class.forName("类名字符串")  (注:类名字符串是包名+类名)  说明:装入类,并做类的静态初始化,返回Class的对象 3.实例对象.getClass()  说明:对类进行静态初始化.非静态初始化:返回引用运行时真正所指的对象(因为:子对象的引用可能会赋给父对象的引用变量中)所