JUC并发编程与源码分析笔记11-Java对象内存布局和对象头

先从阿里及其它大厂面试题说起

  • 你觉得目前面试,你还有那些方面理解的比较好,我没问到的,我说了juc和jvm以及同步锁机制
  • 那先说juc吧,说下aqs的大致流程
  • cas自旋锁,是获取不到锁就一直自旋吗?cas和synchronized区别在哪里,为什么cas好,具体优势在哪里,我说cas避免cpu切换线程的开销,又问我在自旋的这个线程能保证一直占用cpu吗?假如cpu放弃了这个线程,不是还要带来线程再次抢占cpu的开销
  • synchronized底层如何实现的,实现同步的时候用到cas了吗?具体哪里用到了
  • 我说上个问题的时候说到了对象头,问我对象头存储哪些信息,长度是多少位存储

Object object = new Object()谈谈对这句话的理解,一般而言JDK8默认情况下,new一个对象占用多少内存空间

位置所在:JVM里的堆:新生区:伊甸园区。
构成布局:对象头+对象体

对象在堆内存中布局

权威定义

在HotSpot虚拟机里,对象在堆内存中的存储布局可以划分为三个部分:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。

对象在堆内存中存储布局

对象头

对象标记Mark Word

默认存储对象的HashCode、分代年龄和锁标志位等信息,这些信息都是与对象自身定义无关的数据,所以MarkWord被设计成一个非固定的数据结构以便在极小的空间内存存储尽量多的数据。它会根据对象的状态复用自己的存储空间,也就是说在运行期间MarkWord里存储的数据会随着锁标志位的变化而变化。

存储内容标志位状态
对象哈希码、对象分代年龄01未锁定
指向锁记录的指针00轻量级锁定
指向重量级锁的指针10膨胀(重量级锁定)
空,不需要记录信息11GC标记
偏向线程ID、偏向时间戳、对象分代年龄01可偏向
锁状态25bit31bit1bit4bit1bit2bit
锁状态cms_free分代年龄偏向锁锁标志位
无锁unusedhashCode001
偏向锁ThreadID(54bit) Epoch(2bit)ThreadID(54bit) Epoch(2bit)101
类元信息(类型指针)

存储指向该对象类元数据的首地址,用于找到类元信息,虚拟机通过这个指针确定这个对象是哪个类的实例

对象长度

在64位操作系统中,Mark Word占8个字节,类型指针占了8个字节,一共是16个字节。

实例数据

存放类的属性数据信息,包括父类属性。

对齐填充

虚拟机要求对象起始地址必须是8字节的整数倍,方便JVM寻址。

官网理论

OpenJDK
OpenJDK

再说对象头的Mark Word

在这里插入图片描述
对象布局、GC回收和后面的锁升级就是对象标记MarkWord里标志位的变化。

聊聊Object object = new Object()

JOL证明

JOL官网
JOL:Java Object Layout:用于分析对象在JVM的大小和分布。

代码

pom坐标

<dependency>
    <groupId>org.openjdk.jol</groupId>
    <artifactId>jol-core</artifactId>
    <version>0.9</version>
</dependency>
import org.openjdk.jol.info.ClassLayout;
import org.openjdk.jol.vm.VM;

public class JOLDemo {
    public static void main(String[] args) {
        System.out.println(VM.current().details());
        Object object = new Object();// 16 bytes
        System.out.println(ClassLayout.parseInstance(object).toPrintable());
        MyObject myObject = new MyObject();
        System.out.println(ClassLayout.parseInstance(myObject).toPrintable());
    }
}

class MyObject {
    int id;
    String name;
}

GC年龄采用4位bit存储,最大为15,例如MaxTenuringThreshold参数默认值就是15

尝试添加VM options:-XX:MaxTenuringThreshold=16,再次启动程序,发现启动报错了,因为64位虚拟机上,对象分代年龄存储空间是4bit,最大值也就是15。

尾巴参数说明

在cmd窗口里,输入java -XX:+PrintCommandLineFlags -version,可以看到这些信息。

-XX:InitialHeapSize=265024128 -XX:MaxHeapSize=4240386048 -XX:+PrintCommandLineFlags -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:-UseLargePagesIndividualAllocation -XX:+UseParallelGC
java version "1.8.0_231"
Java(TM) SE Runtime Environment (build 1.8.0_231-b11)
Java HotSpot(TM) 64-Bit Server VM (build 25.231-b11, mixed mode)

其中-XX:+UseCompressedClassPointers表示开启了指针压缩。
如果想要关闭指针压缩,可以在VM options里配置:-XX:-UseCompressedClassPointers,表示关闭指针压缩。
在开启指针压缩的时候,类型指针占4个字节,关闭指针压缩后,类型指针占8个字节。
默认配置,开启了指针压缩,一个对象就是8(对象标记)+4(压缩后的类型指针)+4(对齐填充)=16个字节。
手动关闭了指针压缩,一个对象就是8(对象标记)+8(未压缩的类型指针)=16个字节。

换成其他对象试试

使用自定义对象,添加其他类型的属性,查看对象里的内存结构。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值