例子1: 非堆方法
public static void main(){
int x = 1;
show();
}
public static void show(){
int x = 2;
}
Step 1 | main方法入口处,虚拟机栈开始创建一个栈帧压栈,并且创建一个局部变量表存放 x,赋值1 |
Step 2 | JVM执行show方法,再次创建一个栈帧并入栈虚拟机方法栈,在栈内存中再次开辟一个新的内存空间,存放int类型变量x,并赋值为2,此时 main方法栈帧和show方法栈帧都属于线程私有,互不干涉。 |
Step 3 | show方法执行完毕,变量x 释放,show方法出栈,内存释放。 |
例子2:栈堆联合
public static void main(){
int[] x = new int[3];
x[0] = 20;
}
Step1 | main方法压栈,开辟虚拟机栈方法空间,创建 x 数组变量,此时x 无内存地址。 |
| 同时在堆中开辟空间,存放 new int[3] 数组,堆内存会自动创建一个内存地址值:如0x0045,同时把这个地址值赋给 虚拟机栈中的x 变量,x就有了地址值。可以说 x 指向了这个堆数组。此时所有元素都未赋值,但都有初始化值。 |
Step2 | 将20赋值给数组下标为0 的元素,这样数组三个元素分别为 20,0,0 |
如下图:
例子3:堆栈交互+GC
public static void main(){
int[] x = new int[3];
x[0] = 20;
x = null;
}
Step 1 | 同上 |
Step 2 | 执行 x = null,即 x 的引用数组内存地址 0x0045被删除,在java 堆中创建的 数组地址不再指向 虚拟机栈内存中的数组,堆中的数组不再被 x 所使用,视为垃圾,JVM 会启动垃圾回收机制,不定时自动删除。 |
如下图:
例子4:多对象交互+GC 这里不仅仅针对基础数据类型,对应 Java类对象同样适用
public static void main(){
int[] x = new int[3];// Step 1
x[0] = 10; // Step 1
int[] y = x; // Step 2
y[1] = 20; //step 2
x = null;
}
Step 1 | 同上 略 |
Step 2 | 执行 int[] y = x;在虚拟机栈内存中又定义了一个局部变量y,同时将 x 的内存地址值 0x0045 付给 y,此时x y 指向堆中相同的数组。 并且此时 java 堆中的 int数组对象 下标1赋值为20,对于 x 对象也是如此,对应java 对象也是这样。 |
Step 3 | 将 虚拟机栈中 变量 x 的引用java 堆中数组内存地址 0x0045 删除,此时y对象仍然 非空并且有值。而此时堆中的数组因为还有y来引用,所以不会被GC回收。 |
如下图: