Java虚拟机在执行Java程序的过程中会把它管理的内存划分为若干个不同的数据区域。
从线程的角度来看,运行时的数据区域主要分为两部分,第一部分是线程隔离的数据区,第二部分是线程共享的数据区。
线程隔离的数据区分为三部分,下来我们一一介绍:
1.程序计数器
程序计数器是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器,每条线程都有一个独立的程序计数器。
字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。
2.Java虚拟机栈
虚拟机栈描述的是Java方法执行的内存模型,每个方法执行的同时都会创建一个栈帧用于存放局部变量表、操作数栈、动态链接、方法出口等信息。
每一个方法从调用直至执行完成的过程,就对应一个栈帧在虚拟机栈中入栈到出栈的过程。
局部变量表存放了编译期可知的各种基本数据类型、对象引用、return地址。
3.本地方法栈
线程共享的两个区域:
1.Java堆(对象)
Java虚拟机规范中的描述是:
所有的对象实例及数组都在这里分配内存。
其中Java堆是垃圾收集器管理的主要区域,因此很多时候也被称为“GC堆(Garbage Collected Heap)”。现在GC基本都采用分代收集算法,因此Java堆可以分为:新生代和老年代。
2.方法区(类)
它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池,用于存放编译期生成的各种字面量和符号引用,这部分内容在类加载后进入方法区的运行时常量池中存放。
这里可以这样理解常量池与运行时常量池,常量池是class文件定义的,而运行常量池由Java虚拟机加载在内存中。