Synchronized与锁升级

一:java对象内存布局

对象在堆内存的存储布局可以划分为三个部分:对象头(Header)、实例数据(Instance Data) 和对齐填充

二:对象在堆内存中的存储布局

1:对象头 (在64位系统中,Mark Word占了8个字节,类型指针占了8个字节,一共是16个字节)
对象标记(Mark Word):默认存储对象的HashCode、分代年龄和锁标志等信息。
类元信息:对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象哪个类的实例
在这里插入图片描述
2:实例数据
存放类的属性(Field)数据信息,包括父类的属性信息
3:对齐填充
保证8个字节的倍数

在这里插入图片描述

三:Sychronized的锁升级

Synchronized用的锁是存在Java对象头里的MarkWord中,锁升级功能主要依赖MarkWord中锁标志位和释放偏向锁标志位
无锁:
偏向锁:MarkWord存储的是偏向的线程ID
轻量锁:MarkWord存储的是指向线程栈中Lock Record的指针
重量锁:MarkWord存储的是指向堆中的monitor对象(系统互斥量指针)
在这里插入图片描述

四:无锁

在这里插入图片描述

五:偏向锁

1:偏向锁的持有?
单线程竞争,当线程A第一次竞争到锁时,通过修改MarkWord中的偏向线程ID、偏向模式。如果不存在其他线程竞争,那么持有偏向锁的线程将永远不需要进行同步。

2:偏向锁的撤销?
当有另外一个线程逐步来竞争锁的时候,就不能再使用偏向锁了,要升级为轻量级锁,竞争线程尝试CAS更新对象头失败,同时检查持有偏向锁的线程是否还在执行:
(1)第一个线程正在执行Synchronized方法(处于同步块),它还没有执行完,其他线程来抢夺,该偏向锁会被取消掉并出现锁升级,此时轻量级锁由原来持有偏向锁的线程持有,继续执行同步代码块,而正在竞争的线程会自动进入自旋等待获得该轻量级锁
(2)第一个线程执行完Synchronized(退出同步块),则将对象头设置为无所状态并撤销偏向锁,重新偏向。

3:案例说明

public class SynchronizedUpDemo {
   public static void main(String[] args) {
       /**
        * 这里偏向锁在JDK6以上默认开启,开启后程序启动几秒后才会被激活,可以通过JVM参数来关闭延迟 -XX:BiasedLockingStartupDelay=0
        */
        try { 
	        TimeUnit.SECONDS.sleep(5); } catch (InterruptedException e) { e.printStackTrace(); }
	        o = new Object();
	       synchronized (o) {
	           System.out.println(ClassLayout.parseInstance(o).toPrintable());
          }
   }
}

在这里插入图片描述

六:轻量级锁

1:什么是轻量级锁?
多线程竞争,但是任意时候最多只有一个线程竞争,即不存在锁竞争太激烈的情况,也就没有线程阻塞。

在这里插入图片描述
2:轻量级锁升级到重量级锁
自旋一定程度和次数(Java8 之后是自适应自旋锁------意味着自旋的次数不是固定不变的
(1)线程如果自旋成功了,那下次自旋的最大次数会增加,因为JVM认为既然上次成功了,那么这一次也大概率会成功
(2)如果很少会自选成功,那么下次会减少自旋的次数甚至不自旋,避免CPU空转

七:重量级锁

1:有大量线程参与锁的竞争,冲突性很高
2:重量级锁原理
在这里插入图片描述
3:代码演示
在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值