1. 程序计数器
线程私有,可以看做是当前线程执行的字节码的行号指示器,字节码解释器的工作就是通过改变程序计数器的值来选取下一条要执行的字节码。如果一个线程在执行的方法是Native方法则程序计数器的值为null,是唯一一个没有规定任何OOM情况的区域
2. 虚拟机栈
线程私有,生命周期与线程相同,每个方法执行的时候,虚拟机都会为方法创建一个栈帧将其压入虚拟机栈,栈帧包括局部变量表,操作数栈,动态链接,方法出口,等其他信息,其中局部变量表存放编译器可知的各种java虚拟机的基本数据类型,对象引用等,局部遍链表的存储空间及局部变量槽位的形式来体现,一般在编译时期就确定,在运行时期不可改变。可能出现StackOverFlowError,栈内存的OOM发生在创建线程的时候内存不足无法分配,而出现OOM在运行时候不会出现。
3. 本地方法栈
为虚拟机执行java的本地方法服务
4. 堆
线程共享,是内存管理的主要区域,现代大多数的垃圾收集器都是基于分代理论,所以会出现新生代,老年代,Eden空间,From区域,To区域,当线程在堆中分配对象的时候是分配在线程私有的TLAB缓冲区域java堆可以通过虚拟机参数-Xms和-Xmx来设置,当java堆无法扩展,实例不能分配的时候会出现OOMError
5. 方法区
也叫非堆,主要是为了区别与堆内存的区别,方法区主要存放的是一些加载的类型信息,常量,静态变量等。注意以前永久代的概念就是方法区的实现使得垃圾收集器像管理堆内存一样管理方法区,但是这样更容易遇到内存溢出的问题,内存大小受限于JVM对运行时数据区的内存分配,在JDK8时候将永久代的实现理念替换为元空间,使用计算机本身的内存。这部分也有垃圾回收,主要针对的是常量池的垃圾回收和类型卸载
运行时常量池是方法区的一部分.
直接内存:jdk4的时候出现NIO基于通道与缓冲区的I/O方式,操作堆外内存通过存储在堆里面的一个对象引用对这块内存进行操作。
6. 对象的创建
6.1 加载类
6.2 分配内存
内存分配的方式主要有指针碰撞( 内存规整)或者空闲列表而内存是否规整取决于垃圾回收器回收内存之后是否会进行空间压缩整理
6.3 对分配的内存空间进行初始化(不包括对象头)
6.4 对对象头进行设置,对象头主要包括Mark Work和类型指针,以及对其填充,如果是数组类型的话还有数据长度。Mark Work主要包括 对象的hashcode 分代年龄,元数据信息以及相关锁信息。
6.5 执行<init>方法包括 执行构造函数,实例代码块等
7. 对象的访问定位
有句柄和直接引用两种,使用句柄的话会在堆内存中划分出句柄池,栈中引用存储的就是句柄信息,直接引用的话就是直接指向堆中对象的引用
使用指针访问更快,而使用句柄的话更为稳定