一. 线程私有
1. 程序计数器( Program Counter Register)
当前线程所执行字节码的行号。
2. Java虚拟机栈( Java Virtual Machine Stacks)
Java方法执行的内存模型: 每个方法在执行的同时都会创建一个栈帧( Stack Frame) 用于存储局部变量表、 操作数栈、 动态链接、 方法出口等信息。 每一个方法从调用直至执行完成的过程, 就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。
在Java虚拟机规范中, 对这个区域规定了两种异常状况: 如果线程请求的栈深度大于虚拟机所允许的深度, 将抛出StackOverflowError异常; 如果虚拟机栈可以动态扩展( 当前大部分的Java虚拟机都可动态扩展, 只不过Java虚拟机规范中也允许固定长度的虚拟机栈) , 如果扩展时无法申请到足够的内存, 就会抛出OutOfMemoryError异常。
3. 本地方法栈( Native Method Stack)
本地方法栈与虚拟机栈所发挥的作用是非常相似的, 它们之间的区别不过是虚拟机栈为虚拟机执行Java方法( 也就是字节码) 服务, 而本地方法栈则为虚拟机使用到的Native方法服务。
二. 线程共有
4. Java堆( Java Heap)
虚拟机启动时创建。 此内存区域的唯一目的就是存放对象实例, 几乎所有的对象实例都在这里分配内存。
Java堆是垃圾收集器管理的主要区域,因此也叫(GarbageCollected Heap)。
从内存回收的角度来看,Java堆中还可以细分为: 新生代,老年代。
5. 方法区( Method Area)
存储已被虚拟机加载的类信息、 常量、 静态变量、 即时编译器编译后的代码等数据。
Java虚拟机规范把方法区描述为堆的一个逻辑部分。
垃圾收集器也管理方法区,也叫永久代。HotSpot虚拟机, 根据官方发布的路线图信息, 现在也有放弃永久代并逐步改为采用Native Memory来实现方法区的规划。
5.1 运行时常量池( Runtime Constant Pool)
方法区的一部分。 Class文件中除了有类的版本、 字段、 方法、 接口等描述信息外, 还有一项信息是常量池( Constant Pool Table) , 用于存放编译期生成的各种字面量和符号引用, 这部分内容将在类加载后进入方法区的运行时常量池中存放。
6. 直接内存( Direct Memory)
不是虚拟机运行时数据区的一部分, 也不是Java虚拟机规范中定义的内存区域。
本机直接内存的分配不会受到Java堆大小的限制, 但是, 既然是内存, 肯定还是会受到本机总内存( 包括RAM以及SWAP区或者分页文件) 大小以及处理器寻址空间的限制。也可能导致OutOfMemoryError异常出现。