总体划分
Java种由JVM来管理内存的分配和回收,JVM将内存区域总体可以分为堆区和非堆区,非堆区又分为JVM栈,方法区,本地方法栈,程序计数器等。其中堆区和方法区为所有线程共享的数据区,其他为各个线程私有的数据区。
Java堆
Java堆是JVM所管理的最大的一块区域了,也是我们最关心的。几乎所有的对象实例都在这里分配。(JIT与逃逸分析技术使得对象也可以在栈上分配)
Java堆是垃圾收集器管理的主要区域,所以也被称为GC堆。从内存回收的角度看(分代收集算法),Java堆可以细分为:新生代和老年代,更细一点有:Eden空间、From Survivor空间、To Survivor空间等。从内存分配的角度看,Java堆可划分出多个线程私有的分配缓冲区(TLAB),主要用于小对象的分配。Java堆在物理空间可以不连续,只要逻辑上是连续的就可以。
关于TLAB:
由于对象一般会分配在堆上,而堆是全局共享的。因此在同一时间,可能会有多个线程在堆上申请空间。因此,每次对象分配都必须要进行同步,而在竞争激烈的场合分配的效率又会进一步下降。JVM使用TLAB来避免多线程冲突,每个线程使用自己的TLAB,这样就保证了不使用同步,提高了对象分配的效率。
TLAB本身占用eden区空间,TLAB空间的内存非常小,缺省情况下仅占有整个Eden空间的1%,由于TLAB空间一般不会很大,因此大对象无法在TLAB上进行分配,总是会直接分配在堆上。
方法区
方法区也是线程共享的区域,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码、运行时常量池等。
运行时常量池:Class文件常量池(字面量和符号引用)种的内容在类加载后进入方法区种的运行时常量池中存放。(Class文件中除了包括类的版本、字段、方法、接口等描述信息外,还包括常量池)
哪些是Class文件常量池中的内容:
- 字面量:文本字符串、被声明为final的常量值、基本数据类型的值等
- 符号引用:类和接口的完全限定名、字段和名称的描述符、方法名称和描述符
参考:
JVM栈
JVM栈是线程私有的,生命周期与线程相同。Java中的方法在执行时都会创建一个栈帧,用于存储局部变量表,操作数栈,方法出口等信息,每一个方法的执行到完成的过程,就对应着一个栈帧在JVM栈中入栈到出栈的过程。如果线程请求的栈深度大于JVM所允许的深度时,就会抛出StackOverflowError异常。栈中保存的时基本数据类型的值和对象的引用地址,实际对象在堆中。
本地方法栈
功能与JVM栈相同,只不过JVM栈是为JVM执行的方法(也就是字节码)服务,而本地放发栈是为JVM使用到的native方法(采用C、C++等其他语言实现)服务。
程序计数器
被看作当前线程所执行的字节码的行号指示器,用来记录当前线程正在执行的指令,是线程私有的。为了线程切换后恢复到正确的位置继续执行,所以每个线程都需要有一个独立的程序计数器。如果执行的是native方法,这个计数器的值为空。
参考:《深入理解Java虚拟机》