Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域,如下图所示:
图片来自网络
运行时数据区域
程序计数器
程序计数器是一块较小的内存空间,线程私有,它的主要特点:
- 作为当前线程所执行的字节码的行号指示器。
- 分支、循环、跳转、异常处理等基础功能都依赖这个计数器。
- 各条线程之间计数器互不影响,独立存储。
- 如果线程正执行的是Java方法,计数器记录的是字节码指令的地址,如果执行的是Native方法,这个计数器值则为空。
Java虚拟机栈
线程私有,生命周期和线程相同,描述Java方法执行的内存模型,主要特点:
- 每个方法在执行的同事创建一个栈帧,用于存储局部变量表、操作数栈、动态链接、方法出入口等,方法的调用到执行完成,对应着栈帧在虚拟机栈的入栈出栈过程。
一般来说,粗略将Java内存分为堆内存和栈内存,所指的栈即是虚拟机栈,或者说虚拟机栈中的局部变量表。
局部变量表存放了编译器可知的基本数据类型(boolean、byte、char、short、int、float、long、double)、对象引用类型(reference)和returnAddress类型(指向字节码指令的地址)。
64位长度的long和double类型的数据会占用2个局部变量空间,其余的数据类型只占用1个,局部变量表所需的内存空间在编译器内完成分配,方法在帧中分配多大的局部变量空间是确定的,方法运行期间不会改变局部变量表的大小。
线程请求的栈深度大于虚拟机所允许的深度将会抛StackOverflowError异常;如果虚拟机栈可以动态扩展,扩展时无法申请到足够的内存,就会抛OutOfMemoryError异常。
本地方法栈
本地方法栈和虚拟机栈作用相似,区别在于本地方法栈执行的是Native方法服务。
Java堆
所有线程共享的内存区域,该内存唯一的目的就是存放对象实例及数组,几乎的对象实例都在这里分配内存。虽然Java虚拟机的规范描述是所有的对象实例和数组都要在对上分配,但是随着JIT编译器的发展,这也不是那么绝对的事了。
由于对象实例都在堆分配内存,所以垃圾收集管理器的主要区域就是在此。
方法区
所有线程共享的内存区域,用于存储被虚拟机加载的类信息、常量、静态变量、即时编辑器编译后的代码等数据。
运行时常量池
是方法区的一部分,用于存放编译器生成的各种字面量和符号引用,这部分内容将在类加载存放在常量池中。
网友评论