3.1 类加载内存分析
当程序主动使用某个类的时候,如果该类还未被加载到内存中,则系统会通过如下三个步骤来对该类进行初始化。
- 加载:将 class 文件字节码内容 加载到 内存中,它们现在还属于是静态的数据。所以我们就把这些静态数据转换成方法区的运行时数据结构,然后生成一个 代表 这个类 的 Class 对象。
也就是说 class 不是我们创建的,而是早已被创建好的,我们只是把它获取过来,进行调用而已!!!
- 链接:将 Java 类的二进制代码 合并到 JVM 的运行状态之中的过程。
验证:确保加载的类信息符合 JVM 规范,没有安全方面的问题。
准备:正式为类变量(static)分配内存并设置类变量 默认初始值的阶段,这些内存都将 在 方法区中进行分配。(所以 static 的成员变量 和 方法 是很早就被 分配诞生的!)
解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址) 的过程。(常量就是个标识符,直接替换为相应的数据就行了。)
- 初始化:
执行构造器<clinit>()方法的过程。类构造器<clinit>()方法是由编译期自动收集类中所有类变量的赋值动作和静态代码块中的语句合并产生的。(类构造器是构造类信息的,不是构造该类对象的构造器)。
当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先将其父类进行初始化,才能再进行自身的初始化。
虚拟机会保证 一个类的 <clinit>() 方法在多线程环境中被正确加锁和同步。
- 类加载内存分析代码
package www.muquanyu.reflection;
public class 类加载内存分析 {
public static void main(String[] args) {
A a = new A();
System.out.println(A.m);
}
}
class A {
static {
System.out.println("A类静态代码块初始化");
m = 300;
}
static int m = 100;
public A() {
System.out.println("A类无参构造初始化");
}
}
由方法区 类加载 在堆区创建了一个 class 对象,然后进行链接会对常量和一些成员变量进行默认值赋值。再进行 类初始化 会合并代码,比如说static静态代码块的m=300和static int m =100 就会发生合并成 m = 100。合并后,我们再去根据 这个 class 对象 去实例化。实例化的时候 会分配 该对象的 在堆区的内存地址到 栈区里存储起来,以便我们找到 这个 实例化对象。