1. Java内存分析
下面聊完类的加载,再以一个实例来解析这个内存
2. 类的加载过程
当程序主动使用某个类时,如果类还未加载到内存中,则系统会通过如下三个步骤对该类进行初始化
类的加载与ClassLoader的理解
【1】 加载: 将class文件字节码内容加载到内存中,并将这些静态数据转化成方法区的运行时数据结构,然后生成一个代表这个类的java.lang.Class对象
【2】链接:将java类的二进制代码合并到JVM运行状态之中的过程
- 验证:确保加载的类信息符合JVM规范,没有安全方面的问题
- 准备:正式为类变量(static) 分配内存并设置遍历默认初始值阶段,这些内存都将在方法区中进行分配
- 解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程
【3】初始化:
- 执行类构造器()方法的过程。类构造器()方法是由编译器主动收集类中所有类变量的赋值动作和静态代码块中的语句合并生成的。(类构造器是构造类信息的,不是构造该类对象的构造器)
- 当初始化一个类的时候,如果发现有其父类还没有初始化,则需要先触发父类的初始化
- 虚拟机会保证一个类的clint() 方法在多线程环境中被正确加锁和同步
上面一大堆理论后,我们还是通过一个具体的实例来了解一下这个过程
package com.gs.reflection;
//测试类的加载
public class Test05 {
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类的无参构造器初始化");
}
}
上面代码的结果:
对结果进行分析:
- 加载:代码运行时,先把类A和Test05的class文件字节码内容加载到内存,并把相关的数据放在方法区;最后他们会在堆中生成相应的java.lang.Class类。
- 链接,为静态代码块赋初值,为常量赋直接引用 =>这里对应上面的代码时就是将静态变量m覆初值为0
- 初始化,我们从上面的解析中可以看到它是先把所有类变量的赋值动作和静态代码块的语句合并执行,然后再进行初始化。
所以它的执行顺序为:
[1] clinit(){
System.out.println(“A类静态代码初始化”);
m = 300;
m = 100;
}
[2] A类的初始化
相关的内存分析图
以上内容参考自狂神说Java中的注解和反射