加载
我们的字节码文件都是加载到方法区中,而方法区在1.8版本以后就是使用元空间来实现的。而这个文件包括了些什么呢?就是以下(这些东西都是用C++的instanceKlass来描述Java类的)
存储在方法区的都是instanceKlass,而我们注意到java_mirror(类镜像),这个才是我们真正的平常时候使用.class所会引用到的对象(java_mirror存储着该instanceKlass在Java堆中.class对象的地址)。
我们使用反射的时候会利用对象头里记录的class地址找到堆中的.class文件,然后这个java_mirror镜像类可以帮我们找到元空间里面的字节码文件,从而获得各种信息。
链接
- 验证:检查是否符合JVM规范,以及安全性的检查(比如你改个Magic Value cafe babe)
- 准备:为静态变量分配空间,赋予默认值。(静态变量1.8以后存储在Java_Mirror堆中的类镜像里了,以前是存在方法区的instanceKlass里的)
- 解析:将常量池中的符号引用替换为直接引用。
我们可以看到UnresolvedClass这里,未解析之前并没有标明具体的地址,但是如果我们真正完成解析之后,我们就可以看到地址就出现了。
初始化
类初始化是懒惰的,也就是说是懒汉式的,用到的时候才会进行一个加载。那么什么叫用到的时候呢?比如说
- main方法所在类总是会被先初始化
- Class.forName或者是new()。
- 你子类用到了,但是父类还没有初始化。那么父类也会被初始化。
- 首次访问静态变量和方法的时候。
访问.class不会初始化,访问static final也不会,类加载器的loadClass也不会触发初始化,创建该类的数组也不会初始化。
初始化了之后就会赋予变量我们所设置的值了,而不是默认值。