1、程序计数器(Program Counter Register)
这是一块比较小的空间,可以看作是当前线程所执行的字节码的行号指示器。字节码解释器的工作就是通过改变计数器的值来选取 下一条需要执行的字节码指令,分支、循环、跳转、异常处理等功能都需要依赖这个计数器。由于JVM的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的。所以每条线程都需要有一个独立的PCR,所以称这类区域为“线程私有空间”。这是惟一一个没有规定Out of MemoryError的区域。
2、java虚拟机栈(Java Virtual Machine Stacks)
此区域也是线程私有的,生命周期与线程相同。此区域描述的是java方法的内存模型。每个方法在存储的过程中都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。方法的调用到执行完成的过程,对应入栈和出栈的过程。
局部变量表存储编译期可知的各种基本数据类型(boolean byte int char short float double long)、对象引用类型,可能是一个指向对象起始地址引用指针,也可能是指向一个代表对象的句柄或其他与此对象相关的return Address(指向字节码指令的地址)。
此区域会出现两种异常:Stack OverflowError和Out of Memory异常。
3、本地方法栈(Native Method Stack)
此区域用来执行虚拟机使用到的Native方法。也会抛出两种异常:Stack OverflowError和Out of Memory异常。
4、java 堆(Heap)
这是虚拟机所管理的最大的一块区域,也是被所有线程共享的 一片区域,VM启动时,唯一的目的就是存放对象实例,几乎所有的对象都在这里分配内存。
也是垃圾收集器管理的最主要的区域,因此也被称为GC堆,因为有分代收集算法,所以还可以细分为:新生代和老年代;再细致有:Eden空间、FromSurvivor空间、To Suivivor空间等。JVM规定,Heap可以是处于物理上不连续的空间,只要逻辑上是连续的即可。
5、方法区(Method Area)
也是线程共享的区域,用于村粗被JVM加载的类信息、常量、静态比阿亮、即时编译器编译后的代码等数据。别名(Non-Heap)。当方法区无法满足内存分配需求时,将抛出OutofMemory异常。
6、运行时常量池(Runtie Constant Pool)
是方法区的一部分,Class文件中除了有版本、字段、方法、接口等描述信息外,还有一项就是常量池信息,用于存放编译期生成的各种字面量和符号引用,这部分内容在类加载后进入方法区的运行时常量池中存放。也会抛出OutofMemory异常。