对于JVM来说主要包括几部分
- 类装载子系统: 加载 -> 验证 -> 准备 -> 解析 -> 初始化
a. 加载:通过IO读取文件,只有使用到这个类的时候才会加载
b. 验证:检查class文件是否正确
c. 准备:为静态变量分配内存,并且赋默认值
d. 解析:将符号引用替换为直接引用,处理静态方法。符号引用就是指某些静态方法(代号)替换为指向数据所在内存的指针或句柄等(直接引用),这是所谓的静态链接过程(类加载期间完成)。与之相对的还有动态链接(运行期间完成),在栈帧中存在一块内存区域记录动态链接。
e. 初始化:对静态变量赋指定值,执行静态方法块 - 字节码执行引擎
a. 执行方法区上的方法
b. 修改程序计数器 - 运行时数据区
运行时数据区
运行时数据区主要包括
- 堆:对象创建几乎都是堆上。分为新生代:Eden区、S1、S2,老年代
- 方法区:主要存放类元信息、常量、静态变量。JDK8后,方法区存放在直接内存上,所以最大可以占满整个内存
上述是所有线程共享,接下来是每个线程独享
- 栈:由一个个栈帧组成。执行一个方法就会创建一个栈帧
栈帧包含下面几部分- 操作数栈
- 局部变量表
- 动态链接:运行期间将符号引用替换为直接引用
- 方法出口
- 程序计数器(PC寄存器):记录执行位置的内存,线程恢复时根据它进行继续执行,由字节码执行引擎修改。
- 本地方法栈:类似于栈,存放本地(native)方法
一图说
class A{
private static final int i = 1;
private static B b = new B();
public static void main(String[] args){
A a = new A();
a.test();
}
private void test(){
System.out.println(i);
}
}