JVM内部锁升级过程(偏向锁,轻量级锁,重量级锁)

目录

对象在内存中是如何布局的

如何查看对象在内存中的布局

markword数据结构

加锁后发生了什么

偏向锁

 什么是偏向锁

偏向锁定时hashCode 哪去了?

为什么需要偏向锁

为什么从JDK15开始默认关闭偏向锁

什么是安全点

JDK8 为什么要延迟4S后开启偏向锁

锁升级流程

轻量级锁(Thin Lock)

自旋锁

自适应自旋锁

重量级锁(Fat Lock)

Synchronized 锁实现

监视器锁 (monitor)

总结 


对象在内存中是如何布局的

在聊到对象加锁这个话题,那就必须先聊聊对象在内存中的布局, 你知道一个对象在内存中是如何布局的吗?一个对象new出来以后,它在内存中主要分为一下四个部分:

markword 这部分就是加锁的核心,占8个字节
klass pointer 记录指向对象class文件的指针,占4个字节
instance data 对象变量数据
padding 对其数据,在64位版本虚拟机规范中要求对象大小必须是8的倍数,不足部分使用padding补齐
final Object monitor = new Object();  这个monitor 在内存中的大小是多少字节呢?答案是16个字节,8+4+0=12,12不能不8整除,所以补齐后的大小为16;
OFF  SZ   TYPE DESCRIPTION               VALUE
  0   8        (object header: mark)     0x0000000000000005 (biasable; age: 0)
  8   4        (object header: class)    0xf80001e5
 12   4        (object alignment gap)    

如何查看对象在内存中的布局

   我们可以通过JOL(Java Object Layout)来查看对象在内存中的布局;

       <dependency>
            <groupId>org.openjdk.jol</groupId>
            <artifactId>jol-core</artifactId>
            <version>0.16</version>
        </dependency>
public static void main(String[] args) {
         //JKD8延迟4S开启偏向锁
        Thread.sleep(5000);
        
        final Object monitor = new Object();
        System.out.println(ClassLayout.parseInstance(monitor).toPrintable());
}

这个代码的运行结果就是上文中的布局信息。

markword数据结构

       在上文中提到markword是对象加锁的核心,那么这部分数据的结果是什么样子呢?下面就来介绍一下markword数据结构;这也一些大厂面试中经常会遇到的部分。

锁状态 56 bit 1bit 4bit 1bit 2bit
是否偏向锁 锁状态
无锁 unused 25bit hashcode 31bit unused 分代年龄 0 01
偏向锁 线程ID epoch 2bit unused 分代年龄 1 01
轻量级锁 指向战中锁记录的指针 00
重量级锁 指向互斥锁的指针 10
GC 11

加锁后发生了什么

    聊完了JOL以后,我们来看看对象加锁后到底发生了什么?我们通过如下一段代码来看看:

 public static void main(String[] args) {
         final Object obj = new Object();
        System.out.println("启动后对象布局:\n" + ClassLayout.parseInstance(obj).toPrintable());
        //JKD8延迟4S开启偏向锁
        Thread.sleep(5000);
        //可偏向 101
        final Object monitor = new Object();
        System.out.println("延迟5秒后对象布局:\n" + ClassLayout.parseInstance(monitor).toPrintable());
        //偏向锁
        synchronized (monitor) {
            System.out.println("对象加锁后的布局:\n" + ClassLayout.parseInstance(monitor).toPrintable());
        }
        System.out.println("对象释放锁后的布局:\n" + ClassLayout.parseInstance(monitor).toPrintable());
     }

输出的结果是: 

启动后对象布局:
java.lang.Object object internals:
OFF  SZ   TYPE DESCRIPTION               VALUE
  0   8        (object header: mark)     0x0000000000000001 (non-biasable; age: 0)
  8   4        (object header: cl
  • 2
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

陈脩

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值