java虚拟机在执行java程序的时候会把内存划分为若干个不同的数据区域,这些数据区有着各自的用途。主要包括以下几个运行时数据区域。
程序计数器
程序计数器是一块较小的内存空间,它主要用来记录当前线程所执行的字节码的行号。程序的流程控制都需要依赖这个程序计数器。在执行java方法的时候计数器记录的是虚拟机字节码指令的地址,执行本地方法的时候计数器的值则为空。此区域不会出现OutOfMemoryError。
java虚拟机栈
虚拟机栈的声明周期和线程相同,描述的是java方法执行线程的内存模型。每个方法在执行的时候,都会创建一个栈帧来存储局部变量表、操作数栈、动态链接、方法出口等信息。方法被调用时入栈,执行完毕时出栈。
局部变量表存放基本类型、对象引用、returnAddress。这些数据在局部变量表中的存储空间为局部变量槽。
此区域会出现两类异常:
- StackOverflowError: 线程请求的栈神队大于虚拟机允许的深度。
- OutOfMemoryError: 线程申请栈空间失败。
本地方法栈
本地方法栈用于虚拟机调用本地方法,与虚拟机栈相似,Hotspot将二者合二为一。此区域也会跑出两类异常,StackOutOfMemoryError和OutOfMemoryError。
java堆
java堆是虚拟机内存管理的主要区域,此区域是线程共享的,主要用于存放对象实例。java堆中又划分了新生代和老年代,新生代中又划分了Eden和Survivor。垃圾收集器主要作用在这个区域。这里还划分出了多个线程私有的分配缓冲区(TLAB),以提升对象的分配效率。
可以通过参数-Xmx和-Xms来制定堆的大小,当java堆中不能为实例分配内存空间并且无法再扩容的时候会抛出OutOfMemoryError。
方法区(非堆)
方法区(别名非堆)用于存放虚拟机加载的类型信息、常量、静态变量、即时编译器编译后的代码缓存等。这个区域的主要内存回收为常量的回收和类的卸载。
JDK8以前很多人把方法区称为永久带,在JDK8改为了元空间,并且将字符串常量池和静态变量移出了这个区域,元空间里只剩下了类型信息。
在方法区当无法满足新的内存分配的时候也会抛出OutOfMemoryError。