JVM在执行java程序时,会把管理的内存划分为若干个不同的区域,每个区域都有着不同的功能。线程共享区有方法区和堆,而java栈、本地方法栈和程序计数器为线程私有区。
程序计数器,是一块较小的内存空间,它可以看作当前线程所执行的字节码的行号指示器。因为Java是多线程执行的,一个线程可能还未执行完就因为CPU时间片转轮切换到了另外一个线程,在切换回之前线程的时候,需要回到线程上次的执行位置,所以是线程私有的。
Java栈,它的生命周期与线程相同。java栈描述的是java方法执行的内存模型:每个方法在执行时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。在JVM中,栈帧只有两种操作:出栈和入栈。要执行一个方法时,将该方法的栈帧压入栈顶,方法执行完成该栈帧出栈。正在被线程执行的方法称为当前线程方法,而当前方法的栈帧被称为当前栈。
本地方法栈,作用与java栈相似。区别在于java栈为JVM执行java方法服务,而本地方法栈是为JVM使用Native方法服务。
堆,属于线程共享。它在虚拟机启动时创建,是JVM所管理的内存中最大的一块,主要用于存放对象实例,同时堆也是GC进行管理的主要区域。
方法区,主要用于存储已被JVM加载的类信息、常量、静态变量、即时编译器编译之后的代码等数据。常量池也是方法区的一部分,它存放着字面量和符号引用。字面量就是java中常量,比如文本字符串,final修饰的常量等。方法引用则包括类和接口的全限定符,方法名和描述符,字段名和描述符等。
以下为测试代码,用于了解不同区域存放的信息:
public class demo {
public static void main(String[] args) {
//局部变量a存放在java栈中
int a = 5;
System.out.println("Address of a is " + System.identityHashCode(a));
/* 字符串"abc"存放在方法区的常量池中
str1变量存放着字符串“abc"的地址,该地址位于常量池中
str2变量存放String实例的地址,该地址位于堆中
*/
String str1 = "abc";
String str2 = new String("abc");
System.out.println("Address of \"abc\" is " + System.identityHashCode("abc"));
System.out.println("Address of str1 is " + System.identityHashCode(str1));
System.out.println("Address of str2 is " + System.identityHashCode(str2));
}
}
代码运行结果: