看到一个面试题,说说Class.forName() 与 ClassLoader.loadClass()的区别,特意记录一下,方便后续查阅。
    在我们写java代码时,通常用这两种方式来动态加载一个java类,它们是Class.forName() 与 ClassLoader.loadClass()。 但是这两个方法之间也是有一些细微的差别,下面通过写代码实践的方式来看看结果。

一、Class.forName() 方式

(1)理论

    查看Class类代码里面的具体实现可知,实质上这个方法是调用原生的方法如下:
       /** Called after security check for system loader access checks have been made. */
private static native Class<?> forName0(String name, boolean initialize,
ClassLoader loader,
Class<?> caller)throws ClassNotFoundException;
4
 
1
       /** Called after security check for system loader access checks have been made. */
2
    private static native Class<?> forName0(String name, boolean initialize,
3
                                            ClassLoader loader,
4
                                            Class<?> caller)throws ClassNotFoundException;
        形式上类似于Class.forName(name,true,currentLoader)。综上所述,Class.forName 如果调用成功后:
      • 将一个Java类被有效得加载到内存中;
      • 类默认会被初始化(initialize 这个参数传的是true),即执行内部的静态块代码以及保证静态属性被初始化;
      • 默认会使用当前的类加载器来加载对应的类。

(2)代码说明

        下面通过代码来说明
        参考代码:
public class TestClass {
public static void main(String[] args) throws ClassNotFoundException {
System.out.println("Class.forName 方式加载类--->start");
Class.forName("com.jwx.digital.client.AClass").getClass();
System.out.println("Class.forName 方式加载类--->end");
}
} class AClass {
static {
System.out.println("AClass初始化");
System.out.println("=====AClass静态代码快执行=====");
}
}
14
 
1
public class TestClass {
2
    public static void main(String[] args) throws ClassNotFoundException {
3
        System.out.println("Class.forName 方式加载类--->start");
4
        Class.forName("com.jwx.digital.client.AClass").getClass();
5
        System.out.println("Class.forName 方式加载类--->end");
6
    }
7
}
8

9
class AClass {
10
    static {
11
        System.out.println("AClass初始化");
12
        System.out.println("=====AClass静态代码快执行=====");
13
    }
14
}
    运行结果:
Class.forName 方式加载类--->start
AClass初始化
=====AClass静态代码快执行=====
Class.forName 方式加载类--->end
4
 
1
Class.forName 方式加载类--->start
2
AClass初始化
3
=====AClass静态代码快执行=====
4
Class.forName 方式加载类--->end

二、ClassLoader.loadClass方式

(1)理论

        采用这种方式的类加载策略,由于双亲委派模型的存在,最终都会将类的加载任务交付给Bootstrap ClassLoader进行加载。跟踪源代码,最终会调用原生方法:
    // return null if not found
private native Class<?> findBootstrapClass(String name);
2
 
1
    // return null if not found
2
    private native Class<?> findBootstrapClass(String name);
        与此同时,与Class.forName()方式的最本质的不同是,类不会被初始化,只有显式调用才会进行初始化。综上所述,ClassLoader.loadClass 如果调用成功后:
      • 将一个java类加载到内存中;
      • 类不会被初始化,只有在之后被第一次调用时类才会被初始化;因为虚拟机规范规定了,当遇到new、getstatic、putstatic或invokestatic这4条字节码指令是,如果没有进行过初始化则需要先触发初始化。

(2)代码说明

    下面通过代码来说明
        参考代码:
public class TestClass {
public static void main(String[] args) throws Exception {
System.out.println("ClassLoader 方式加载类--->start");
Class<?> aClass = ClassLoader.getSystemClassLoader().loadClass("com.jwx.digital.client.AClass");
System.out.println("ClassLoader 方式加载类--->end");
/* 虽然上面没有进行类的初始化,
但是虚拟机规范规定了,当遇到new、getstatic、putstatic或invokestatic这4条字节码指令是,如果没有进行过初始化则需要先触发初始化。
生成这4条指令的最常见的Java代码场景是:使用new关键字实例化对象的时候、
读取或者设置一个类的静态字段(被final修饰、已在编译期把结果放入常量池的静态字段除外)的时候,以及调用一个类的静态方法的时候。*/
// 因此我们实例化这个class时,就会调用初始化
System.out.println("我要实例化这个类了+++++");
aClass.newInstance();
}
} class AClass {
static {
System.out.println("AClass初始化");
System.out.println("=====AClass静态代码快执行=====");
}
}
21
 
1
public class TestClass {
2
    public static void main(String[] args) throws Exception {
3
        System.out.println("ClassLoader 方式加载类--->start");
4
        Class<?> aClass = ClassLoader.getSystemClassLoader().loadClass("com.jwx.digital.client.AClass");
5
        System.out.println("ClassLoader 方式加载类--->end");
6
        /* 虽然上面没有进行类的初始化,
7
         但是虚拟机规范规定了,当遇到new、getstatic、putstatic或invokestatic这4条字节码指令是,如果没有进行过初始化则需要先触发初始化。
8
         生成这4条指令的最常见的Java代码场景是:使用new关键字实例化对象的时候、
9
         读取或者设置一个类的静态字段(被final修饰、已在编译期把结果放入常量池的静态字段除外)的时候,以及调用一个类的静态方法的时候。*/
10
        // 因此我们实例化这个class时,就会调用初始化
11
        System.out.println("我要实例化这个类了+++++");
12
        aClass.newInstance();
13
    }
14
}
15

16
class AClass {
17
    static {
18
        System.out.println("AClass初始化");
19
        System.out.println("=====AClass静态代码快执行=====");
20
    }
21
}
    运行结果:
ClassLoader 方式加载类--->start
ClassLoader 方式加载类--->end
我要实例化这个类了+++++
AClass初始化
=====AClass静态代码快执行=====
5
 
1
ClassLoader 方式加载类--->start
2
ClassLoader 方式加载类--->end
3
我要实例化这个类了+++++
4
AClass初始化
5
=====AClass静态代码快执行=====

三、总结

    (1)Class.forName的形式加载类时,默认会执行类的初始化
    (2)ClassLoader.loadClass的方式加载类时,类不会被初始化,只有显式调用才会进行初始化。可以用来延迟加载(采用ClassLoader进行懒加载,就不会调用类的静态代码快,实现延迟加载)

四、补充

 当一个类被主动使用时,Java虚拟机就会对其初始化,如下六种情况为主动使用:

    1. 当创建某个类的新实例时(如通过new或者反射,克隆,反序列化等)
    2. 当调用某个类的静态方法时
    3. 当使用某个类或接口的静态字段时
    4. 当调用Java API中的某些反射方法时,比如类Class中的方法,或者java.lang.reflect中的类的方法时
    5. 当初始化某个子类时
    6. 当虚拟机启动某个被标明为启动类的类(即包含main方法的那个类)


Class.forName() 与 ClassLoader.loadClass()的区别的更多相关文章

  1. 015 反射中的 Class.forName() 与 ClassLoader.loadClass() 的区别

    作者:nnngu GitHub:https://github.com/nnngu 博客园:http://www.cnblogs.com/nnngu 简书:https://www.jianshu.com ...

  2. 反射中Class.forName()和ClassLoader.loadClass()的区别

    一 Java类装载过程 装载:通过累的全限定名获取二进制字节流,将二进制字节流转换成方法区中的运行时数据结构,在内存中生成Java.lang.class对象: 链接:执行下面的校验.准备和解析步骤,其 ...

  3. Class.forName和ClassLoader.loadClass的区别

    Class的装载分了三个阶段,loading,linking和initializing,分别定义在The Java Language Specification的12.2,12.3和12.4. Cla ...

  4. 反射中的 Class.forName() 与 ClassLoader.loadClass() 的区别

    在Java中,类加载器把一个类加载进Java虚拟机中,要经过三个步骤来完成:加载.链接和初始化,其中链接又可以分成验证.准备和解析三步,除了解析外,其它步骤是严格按照顺序完成的,各个步骤的主要工作如下 ...

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

    Class的装载分了三个阶段,loading,linking和initializing,分别定义在The Java Language Specification的12.2,12.3和12.4.Clas ...

  6. Class.forName和ClassLoader.loadClass区别(转)

    Java中class是如何加载到JVM中的:1.class加载到JVM中有三个步骤    装载:(loading)找到class对应的字节码文件.    连接:(linking)将对应的字节码文件读入 ...

  7. 简单总结Class.forName("").newinstance()和new()以及classLoader.loadClass("")的区别

    文章目录 背景 三种方法简单介绍 Class.forName("").newinstance()方式 new方式 classLoader.loadClass("" ...

  8. java两种反射的区别 - Class.forName()和ClassLoader.loadClass()

    在理解这两种反射机制之前,需要弄清楚java类的加载机制. 装载:通过类的全限定名获取二进制字节流(二进制的class文件),将二进制字节流转换成方法区中的运行时数据结构,在内存中生成Java.lan ...

  9. Class.forName和ClassLoader.loadClass的比较【转载】

    Class的装载分了三个阶段,loading,linking和initializing,分别定义在The Java Language Specification的12.2,12.3和12.4.Clas ...

随机推荐

  1. flask中的Flask、request、render_temple、redirect和url_for

    学flask也有一个多星期了,对这个web框架也有了一点的了解,梳理一些基础的知识点,还是小白一只,代码写得比较low,若文章有错误的地方欢迎大佬随时指正,代码中被注释掉的代码是关于预防csrf,无视 ...

  2. python27期day12:推导式、内置函数、高阶函数、匿名函数、作业题

    1.推导式:做一些有规律的数据结构 列表推导式: 普通循环模式: [加工后的变量 for 循环] 示例一:print([i for i in range(1,51)]) 结果:[1, 2, 3, 4, ...

  3. 三层交换机DHCP配置实验(基于Cisco模拟器)

    实验设备: 三层交换机一台,主机若干台,直通线若干 实验目的: 实现客户机从DHCP(动态主机配置协议)服务器上获取动态IP地址. 实验步骤: 1.划分VLAN Switch>enable Sw ...

  4. 10.webpack学习使用

    1.什么是Webpack Webpack 是当下最热门的前端资源模块化管理和打包工具,它可以将许多松散耦合的模块按照依赖和规则打包成符合生产环境部署的前端资源.还可以将按需加载的模块进行代码分离,等到 ...

  5. <Tree> 110 124

    110. Balanced Binary Tree 方法是如果我们发现子树不平衡,则不计算具体的深度,而是直接返回-1.那么优化后的方法为:对于每一个节点,我们通过checkDepth方法递归获得左右 ...

  6. SpringBoot中的日志

    默认情况下,Spring Boot会用SLF4J + Logback来记录日志,并用INFO级别输出到控制台. SLF4J,即简单日志门面(Simple Logging Facade for Java ...

  7. [LeetCode] 909. Snakes and Ladders 蛇梯棋

    On an N x N board, the numbers from 1 to N*Nare written boustrophedonically starting from the bottom ...

  8. [LeetCode] 663. Equal Tree Partition 划分等价树

    Given a binary tree with n nodes, your task is to check if it's possible to partition the tree to tw ...

  9. [LeetCode] 464. Can I Win 我能赢吗

    In the "100 game," two players take turns adding, to a running total, any integer from 1.. ...

  10. [LeetCode] 130. Surrounded Regions 包围区域

    Given a 2D board containing 'X' and 'O'(the letter O), capture all regions surrounded by 'X'. A regi ...