loadClass和Class.forName的区别

类加载机制

  1. 类的生命周期
    一个Java类,从.java文件到可以使用到最后使用结束,经历的过程包含:
step 作用 备注
编译 从.java到.class文件,使得可以被虚拟机识别和使用 -
加载 (被虚拟机读入内存) -
验证 验证Class字节流的数据是否遵守JVM的规定 -
准备 正式为类变量(静态变量)分配内存并设置初始值,并非代码中设置的值 -
解析 将常量池中的符号引用解析为直接引用 验证、准备和解析属于连接阶段
初始化 真正执行类中定义的Java代码 -
使用 - -
卸载 - -

加载、验证、准备、初始化和卸载的执行顺序是确定的,解析阶段可能会在初始化之后,这也是Java动态特性的支撑。

  • 加载
    读取class文件的二进制字节流(不问来源,通过类的全限定类名);将二进制流代表的静态存储结构转化为方法区的运行时数据结构;在堆中生成对应的Class对象,作为方法区数据结构的入口。
    什么时候加载一个class文件,JVM规范没有硬性规定,可以添加JVM参数+XX:+TraceClassLoading来查看类的加载
  • 解析
    JVM规范没有规定解析的执行时机,只要求在执行 anewarray,checkcast, getfield,getstatic,instanceof,invokeinterface,invokespecial,invokestatic,invokevirtual,multianewarray,new,putfield,putstatic这13个字节码指令之前,对他们所使用的符号引用要解析完成。
  • 初始化
    初始化时类加载的最后阶段,初始化阶段是执行类构造器<clinit>()方法的过程。
    初始化的执行时机,有四种情况会触发初始化:
  1. 遇到new、getstatic、putstatic或invokestatic指令时,如果相关的类没有初始化,会触发初始化。场景有new创建对象,读写类的静态变量或调用类的静态方法。注意如果是编译时期加入常量池的静态变量(final static 常量),那么这个静态变量与定义它的类已经剥离了关系,这种调用不会触发该类的初始化。
    例如:
public class InitOrder {
    static {
        System.out.println("initOrder init!");
    }

    public static void main(String[] args) {
        System.out.println("before get final static var");
        // Demo.A在编译时加入了常量池,是共享的数据,访问不会触发Demo类的初始化
        int a = Demo.A;
        System.out.println("get final static var end");
        System.out.println("before get static var");
        // 非常量的访问 会触发初始化
        int b  = Demo.a;
        System.out.println("get  static var end");
    }
}

class Demo {
    static int a = 100;
    final static int A = 1000;
    static {
        System.out.println("Demo init");
    }
}
// 结果
// initOrder init!
// before get final static var
// get final static var end
// before get static var
// Demo init
// get  static var end

  1. 使用Java反射机制的时候,如果类没有初始化,会触发初始化。
  2. 初始化一个类时,会先初始化其父类(如果父类没有初始化的话)
  3. JVM启动时,程序的入口类会先初始化
  4. 使用JDK1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果是REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,且这个句柄对应的类没有进行过初始化,则需要先进行初始化。
    有且仅有以上5种场景会触发初始化。

<clinit>() <init>()

  1. <clinit>()方法是由编译器自动收集类中静态变量的赋值语句以及静态代码块中定义的语句合并产生的,且内部的语句的顺序是由定义的顺序决定的,后面的语句可以访问前面定义的变量,反过来是不可以的。即静态代码块只能访问定义在此之前的静态变量。
  2. <clinit>()<init>()是不同的,前者对应的是类,后者对应的是实例,即前一个是Class的构造器(是编译器生成的),后者是实例对象的构造器(也就是我们定义或继承的构造函数)。且虚拟机会保证子类的<clinit>()执行之前,其父类的<clinit>()一定执行完成,无需显式指定。所以第一个执行<clinit>()的类是java.lang.Object;
  3. 因为父类比子类先执行<clinit>(),所以父类的静态变量和静态代码块是先于子类执行的。
  4. 如果一个类或者接口中没有静态变量或静态代码块,编译器可以不生成<clinit>()
  5. 接口中没有静态代码块,但是可以有静态变量。所以可以有<clinit>()的初始化动作,但是接口和类不同之处在于接口不需要先执行父接口的<clinit>()方法。
  6. 虚拟机会保证一个类的<clinit>()方法在多线程环境中能被正确的同步,利用这一点可以实现线程安全的单例模式。

初始化结束后,一个类就可以被正常的使用了。

loadClass

loadClass是抽象类ClassLoader中实现的方法,先不看classLoader的加载机制,看一下loadClass的实现:
源码:

    public Class<?> loadClass(String name) throws ClassNotFoundException {
        return loadClass(name, false);
    }
    protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
             // 省略加载机制实现代码
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

从源码来看,当调用ClassLoader.loadClass()方法时,调用的是loadClass(name,false),注意到第二个参数false,看一下源码就能理解是设置是否resolve,所以loadClass只是加载,不会解析更不会初始化。

Class.forName()

Class.forName()也可以用来加载一个指定类,那它和上面的loadClass有什么不同呢?看一下源码:

    @CallerSensitive
    public static Class<?> forName(String className)
                throws ClassNotFoundException {
        return forName0(className, true,
                        ClassLoader.getClassLoader(Reflection.getCallerClass()));
    }
    /** Called after security checks have been made. */
    private static native Class<?> forName0(String name, boolean initialize,
                                            ClassLoader loader)
        throws ClassNotFoundException;

很清楚可以看出来,第二个参数的true对应的是initialize,就是这个方法加载完类会初始化。
有一点注意一下,这个方法加载类使用的类加载器是调用这个方法的类所使用的类加载器,ClassLoader.getClassLoader(Reflection.getCallerClass())

参考资料:
《深入理解Java虚拟机:JVM高级特性与最佳实践》 周志明 著

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,888评论 6 515
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,677评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,386评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,726评论 1 297
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,729评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,337评论 1 310
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,902评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,807评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,349评论 1 318
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,439评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,567评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,242评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,933评论 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,420评论 0 24
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,531评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,995评论 3 377
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,585评论 2 359

推荐阅读更多精彩内容

  • 前言 在讨论之前,首先要明白一个Java类加载到JVM中经过的三个步骤 装载: 查找和导入类或接口的二进制数据 链...
    walker_liu_fei阅读 3,829评论 0 2
  • JVM体系结构 JVM是一种解释执行class文件的规范技术。 我翻译的中文图: 类装载器子系统 在JVM中负责装...
    zhazhaxin阅读 11,742评论 7 69
  • 搜索的时候看了好几篇文,自己就想记录一遍,加深一下记忆,以下是原文的地址,受益匪浅。blog.csdn.net/n...
  • 代码编译的结果从本地机器转变为字节码,是存储格式发展的一小步,却是编译语言发展的一大步。 概述 类的加载指的是将类...
    Michaelhbjian阅读 374评论 0 1
  • 社会上有太多假象,这个世界上有太多假象,只要你不相信,只要一切都不去当真,你才会不再天真和懦弱,才不会因为至善的正...
    菽宝阅读 117评论 0 0