【尚硅谷周阳--JUC并发编程】【第十一章--Java对象内存布局和对象头】
一、对象在堆内存中布局
1、权威定义(周志明JVM第三版)
- 在HotSpot虚拟机里,对象在堆内存中的存储布局可以分为三个部分:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。
2、对象在堆内存中的存储布局
2.1、对象头
- 对象头分为对象标记(markOop)和类元信息(klassOop),类元信息存储的是指向该对象类元数据(klass)的首地址。
- 在64位系统中,Mark Word占8个字节,类型指针占了8个字节,一共是16个字节。 因此new一个对象,若对象不存在实例数据,那么它默认就是16个字节。
2.1.1、对象标记Mark Word
- 默认存储对象的HashCode、分代年龄和锁标志位等信息。
- 这些信息都是与对象自身定义无关的数据,所以MarkWord被设计成一个非固定的数据结构以便在极小的空间内存存储尽量多的数据。
- 它会根据对象的状态复用自己的存储空间,也就是说在运行期间MarkWord里存储的数据会随着锁标志位的变化而变化。
2.1.2、类元信息(又叫类型指针)
- Type Pointer(类型指针)是对象头中用来指向对象的类元数据的指针。在 HotSpot 虚拟机中,Type Pointer 存储的是对象所属类的元数据指针,通过 Type Pointer 可以快速定位到对象的类信息,包括方法表、字段表等。
- Type Pointer 在对象实例化时被设置,并且在对象的整个生命周期中不会改变,因为对象的类在创建后是不可变的。当虚拟机需要根据对象的类型执行特定的方法或字段访问时,会使用 Type Pointer 来查找对象的类信息,从而实现动态分派和多态性。
2.1.3、对象头多大
class Customer {
}
- 如果一个类,如上所示,没有属性,那么这个对象创建出来后,在64位系统中,Mark Word占了8个字节,类型指针占了8个字节,一共是16个字节
2.2、实例数据
class Customer {
int id;
boolean flag = false;
}
- 存放类的属性(Field)数据信息,包括父类的属性信息,如上所示的id,flag等
- 因此上面的Customer实例对象别创建出来后暂不考虑压缩指针的情况下,对象为16(对象标记+类型指针)+4(int)+1(boolean)+3(对齐填充)=24字节
2.3、对齐填充(保证8个字节的倍数)
- 虚拟机要求对象起始地址必须是8字节的整数倍。填充数据不是必须存在的,仅仅是为了字节对齐,这部分内存按照8字节补充对齐。
二、对象头的Mark Word
1、32位虚拟机
2、64位虚拟机(重要)
- oop.hpp源码
- markOop.hpp源码
- hash:保存对象的哈希码
- age:保存对象的分代年龄
- biased_lock:偏向锁标识位
- lock:锁状态标识位
- JavaThread*:保存持有偏向锁的线程ID
- epoch:保存偏向时间戳
- Mark Word(64位)分布图,对象布局、GC回收和后面的锁升级就是对象标记MarkWord里面标志位的变化
三、聊聊Object obj = new Object()
1、位置所在
- JVM堆->新生区->eden区
- Object object = new Object()
- Object 是一个模板,存在与方法区
- object 实例对象的引用,存放在栈中
- new Object()实例对象,存放在堆中
2、JOL(Java Object Layout)
- JOL----Java对象布局,官网
- pom.xml
<!--
官网https://openjdk.org/projects/code-tools/jol/
定位:分析对象在JVM的大小和分布
-->
<dependency>
<groupId>org.openjdk.jol</groupId>
<artifactId>jol-core</artifactId>
<version>0.9</version>
</dependency>
public class JOLDemo {
public static void main(String[] args) {
// VM的细节详细情况
System.out.println(VM.current().details());
// 所有的对象分配的字节都是8的整数倍
System.out.println(VM.current().objectAlignment());
}
}
3、代码结果呈现说明
- 只有对象头,没有其他实例数据
public class JOLDemo {
public static void main(String[] args) {
Object o = new Object();
System.out.println(ClassLayout.parseInstance(o).toPrintable());
}
}
- 带有实例数据情况
class Customers {
int id;
boolean flag = false;
}
public class JOLDemo {
public static void main(String[] args) {
Customers c1 = new Customers();
System.out.println(ClassLayout.parseInstance(c1).toPrintable());
}
}
4、为什么GC年龄最大15?
- 由上图可知,GC年龄采用4位bit存储,最大位为1111,即15.
- 可以尝试设置maxTenuringThreshold参数检验,当没有设置时,它的默认值就是15。下面为将值设置为16的结果。
5、压缩指针相关说明
- 打印命令行
## 打印命令行标志
java -XX:+PrintCommandLineFlags -version
- 默认开启指针压缩说明
## 默认开启指针压缩
-XX:+UseCompressedClassPointers
因此在上述代码结果呈现说明中,类型指针只有4位而非8位,就是因为启用了指针压缩
- 上述表示开启了类型指针的压缩,以节约空间,假如不加压缩