运行时数据区域
1.程序计数器:线程私有内存,为字节码解释器指向一条需要执行的指令,是唯一一个没有规定任何OutOfMemoryError的区域。
2.Java虚拟机栈:线程私有内存,为java方法创建栈帧,栈帧用于存储局部变量表、动态链接、方法出口等信息。其中,除了long和double类型会占用两个slot外,其余数据类型均占用一个slot。
3.本地方法栈:线程私有内存,作用同Java虚拟机栈,用于存储本地方法。
4.Java堆:线程共享内存,用于存储对象实例,GC的主要场所。从内存回收角度来看,Java堆可细分为新生代和老年代;从内存分配角度来看,Java堆可划分出多个线程私有的分配缓冲区(TLAB),TLAB用于解决并发情况下的线程安全问题。
5.方法区:线程共享内存,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。由于该区域GC条件苛刻,又被称为“永久带”。方法区包含运行时常量池,用于存储字面量和符号引用。
HotSpot虚拟机对象
对象创建过程
虚拟机遇到一条new指令时,根据指令参数是否在常量池中定位到类的符号引用,判断类是否加载,完成加载后,在java堆中为对象分配空间。如果堆中内存绝对规整,采用“指针碰撞”的分配方法,利用分界指针的移动来分配空间,否则采用“空闲列表”分配方法进行分配。空闲列表记录哪些内存快是可用的。之后将划分的内存初始化为0,设置对象头等信息。new指令结束后会执行方法初始化真实值。
对象的内存布局
- 对象头
对象头一部分用于存储对象自身的运行时数据,如哈希码、GC分代年龄、锁状态标志、线程持有的锁等。另一部分用于存储类型指针,虚拟机通过该指针确定对象是哪个类的实例 - 实例数据
对象真正存储的有效信息 - 对齐填充
虚拟机要求对象的大小必须是8字节的整数倍,对齐填充仅起占位符的作用
对象的访问定位
Java程序需要通过站上的reference数据来操作堆上的具体对象,主流的访问方式有使用句柄和直接指针两种
- 使用句柄访问
jvm堆中会有一块内存作为句柄池,变量引用的地址就是对象在句柄池中的句柄地址。好处是变量引用的地址将是稳定的句柄地址,当对象在移动时,只会改变句柄池中的实例数据指针,而变量引用的地址不会修改。
- 直接指针访问方式
变量引用的地址就是对象在堆中的地址。好处是节省了指针定位开销,速度快。