Class.forName(className)与ClassLoader().loadClass(className)的区别

面试中会遇到"Class.forName(className)与ClassLoader().loadClass(className)的区别",如果需要知道答案,前提需要先了解到类的加载过程,如下图:

1、加载

    类的加载阶段,主要是获取定义此类的二进制字节流,并将这个字节流所代表的静态存储结构转化为方法区的运行时数据结       构,最后在Java堆中生成一个代表这个类的java.lang.Class对象作为方法区这些数据的访问入口。相对于类加载过程的其他    阶段,加载阶段是开发期可控性最强的阶段。我们可以通过定制不通的类加载器,也就是ClassLoader来控制二进制字节流的获取方式。

2、验证

    验证,准备和解析其实都属于连接阶段,而验证就是连接阶段的第一步。这一阶段主要是为了确保Class文件的字节流中包含    的信息复合当前虚拟机的要求,并且不会危害虚拟机自身的安全。主要验证过程包括:文件格式验证,元数据验证,字节码验    证以及符号引用验证。 
   
3、准备

    准备阶段正式为类变量分配内存并设置初始值。这里的初始值并不是初始化的值,而是数据类型的默认零值。这里提到的类变量是被static修饰的变量,而不是实例变量。关于准备阶段为类变量设置零值的唯一例外就是当这个类变量同时也被final修饰,那么在编译时,就会直接为这个常量赋上目标值。

4、解析

  解析时虚拟机将常量池中的符号引用替换为直接引用的过程。

5、初始化

  在准备阶段,变量已经赋过一次系统要求的初始值,在初始化阶段,则是根据程序员通过程序的主观计划区初始化类变量和其   他资源。Java虚拟机规范规定了有4种情况必须立即对类进行初始化(加载,验证,准备必须在此之前完成) 
  1)当使用new关键字实例化对象时,当读取或者设置一个类的静态字段(被final修饰的除外)时,以及当调用一个类的静态     方法时(比如构造方法就是静态方法),如果类未初始化,则需先初始化。 
  2)通过反射机制对类进行调用时,如果类未初始化,则需先初始化。 
  3)当初始化一个类时,如果其父类未初始化,先初始化父类。 
  4)用户指定的执行主类(含main方法的那个类)在虚拟机启动时会先被初始化。

 除了上面这4种方式,所有引用类的方式都不会触发初始化,称为被动引用。如:通过子类引用父类的静态字段,不会导致子类  初始化;通过数组定义来引用类,不会触发此类的初始化;引用类的静态常量不会触发定义常量的类的初始化,因为常量在编  译阶段已经被放到常量池中了。

那么Class.forName(className)与ClassLoader().loadClass(className)的具体区别就是:

  • ClassLoader.loadClass(className)方法,内部实际调用的方法是  ClassLoader.loadClass(className,false);
  • 第2个 boolean参数,表示目标对象是否进行链接,false表示不进行链接,由上面介绍可以,不进行链接意味着不进行包括初始化等一些列步骤,那么静态块和静态对象就不会得到执行
  • ----------------------------------------------------------------------------------->
  • Class.forName(className)方法,内部实际调用的方法是  Class.forName(className,true,classloader);
  • 第2个boolean参数表示类是否需要初始化,  Class.forName(className)默认是需要初始化。
  • 一旦初始化,就会触发目标对象的 static块代码执行,static参数也也会被再次初始化。

测试代码如下:

classLoadTest.java
package classloader;

public class classLoadTest {
    static {
        System.out.println("this is a test!");
    }
}
Classloader.java
package classloader;

public class Classloader {
    public static void main(String[] args) throws ClassNotFoundException {
        
     /* 执行此注释块的代码会打印出:this is a static code!
        try {
            Class.forName("classloader.classLoadTest");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }*/
        
        //执行如下代码无任何输出
        Classloader.class.getClassLoader().loadClass("classloader.classLoadTest");
    }
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值