美文网首页
多线程篇-synchronized解析-偏向锁、轻量级锁、重量级

多线程篇-synchronized解析-偏向锁、轻量级锁、重量级

作者: mg驿站 | 来源:发表于2020-05-26 22:16 被阅读0次

在多线程的使用场景中,synchronized加锁是很重要的机制,一起了解下吧

1、synchronized执行过程中什么情况下是偏向锁、轻量级锁、重量级锁?

2、synchronized加锁过程中发生了什么?

3、什么条件下会触发锁膨胀?

一、基础概念

先一起了解几个基本概念

1、Mark Word

java对象头的Mark Word中存储了HashCode、分代年龄、锁状态等信息,来看下Mark Word的结构,如下图

2、栈帧(stack frame)

方法执行时,在jvm的栈中会创建一个栈帧用来存储局部变量、操作数栈、动态链接、方法出口等信息。方法从调用到执行完成,就是栈帧在虚拟机栈中入栈到出栈的过程。(所以代码块中的局部变量可以实现入栈创建,出栈销毁)

线程中的许多方法同时处于执行状态,对执行引擎来说,活动线程中,栈顶的栈帧才是有效的,称为当前栈帧,与这个栈帧关联的方法称为当前方法。

二、初识synchronized

synchronized是一直在java的加锁中扮演重要角色,都说它是重量级锁,不过这是JDK5的版本的老黄历了,JDK6以后做了了优化,根据不同情况分别使用向锁、轻量级锁和重量级锁了。

JVM通过Monoitor来实现的synchronized的加锁,monitorenter在编译后会加到同步代码块的开始位置,monitorexit加到结束和异常的位置。

让用javap反编译一下接下来的代码代码

Object m_lock = new Object();

  public void demo()

  {

     synchronized(m_lock)

     {

       int i=0;

     }

}

执行javap –c Demo后如下图

在第6行插入了monitorenter,15行插入了monitorexit,来保证代码块同步。

初步了解了synchronized,在什么情况下Synchronized是偏向锁、轻量级锁和重量级锁,以及锁的膨胀

三、偏向锁、轻量级锁和重量级锁的使用和锁膨胀

加锁是因为多个线程竞争临界资源,只有一个线程竞争、两个线程去竞争、n多个线程竞争的激烈程度是不同的。竞争越激烈的情况下,获取锁的代价越大,所以为了减少性能消耗,jvm根据不同竞争情况,将锁分为偏向锁、轻量级锁、重量级锁。

偏向锁、轻量级锁和重量级锁是怎么加锁的呢 ?

1、偏向锁

偏向锁是三种锁中加锁消耗最小的。

HotSpot作者认为大多数情况下,锁不存在竞争关系,总是会被一个线程持有,为了减少互斥锁的代价

偏向锁是怎么加锁成功呢,将Mark Word中的线程Id标记为当前线程id,就加锁成功了,具体的流程如下:

先读取对象的Mark Word 判断是否处于可偏向的状态,即检查Mark Word中的 是偏向状态和锁标志位。

如果可偏向的,说明当前线程可加锁,那么就用CAS操作去将线程id写入到Mark Word中,如果获取锁成功,执行同步代码。如果CAS操作失败,说明其他线程在竞争,并取到了偏向锁,那么等待全局安全点(GC运行之前所有线程需要在安全点阻塞,这就GC过程中常说的Stop The World),将偏向状态改为0,验证已获取锁的线程是否存活,如果死亡,将锁标志位恢复到无锁状态,重新加锁。如果存活,将锁标志位升级为轻量级锁(01)。

如果不是可偏向的,首先验证Mark Word中线程id是否为当前线程,如果是继续执行代码。如果不是,说明锁对象偏向其他线程,等待安全点,验证是否需要升级为轻量级锁。

这就是偏向锁的整个加锁过程了。

偏向锁的锁状态在线程结束后,不会被置为无锁状态,只有在新的线程来获取锁的时候,在安全点设置为无锁状态或者升级为轻量级锁。

这个过程可以参考下图(借用高并发编程艺术的图):

2、轻量级锁

轻量级锁采用CAS自旋锁的方式来完成加锁,相对于重量级锁加锁的代价相对小一些,如果一直获取不到锁状态,自旋占用的资源会超过重量级锁,所以轻量级锁膨胀为重量级锁的条件就是自旋达超过一定次数(默认为10,可以修改PreBlockSpin参数调整)。

轻量级锁的加锁流程如下:

执行同步代码块之前,JVM会在线程的栈帧中创建一个存储锁记录的空间(Lock Record),并将Mark Word拷贝复制到锁记录中(因为已经脱离了原始的Mark Word,官方以displaced 作为前缀,即Displaced Mark Word(置换标记字))。然后尝试通过CAS将Mark Word中的锁记录的指针,指向创建的Lock Record。如果成功表示获取锁状态成功,如果失败,则进入自旋获取锁状态。

可以参考下图(借用高并发编程艺术的图):

3、重量级锁

重量级锁的加锁需要通过mutex和condition variable(个人理解condition variable提供了wait和notify来阻塞线程)来实现的。

重量级锁会让抢占锁的线程从用户态转变为内核态,开销很大。

详细的过程如下图:

注意 :锁膨胀这个过程,只允许升级,不允许降级,即只能偏向锁升级为轻量级锁,轻量级锁升级为重量级锁,不能反过来重量级锁降级为轻量级锁。

文章到这里就结束了,有帮助的话给个留个赞,有问题可以留言。

相关文章

  • Java中的偏向锁、轻量级锁与重量级锁(synchronized

    Java中的偏向锁、轻量级锁与重量级锁(synchronized) 在多线程并发编程中synchronized一直...

  • java synchronized原理导图

    当线程遇到synchronized后的执行流程,由偏向锁->轻量级锁->重量级锁 偏向锁、轻量级锁、重量级锁的上锁过程

  • JUC框架说明

    synchronized 锁升级:偏向锁 -》轻量级锁 -》重量级锁 1、同步状态的原子性管理 ...

  • synchronized锁升级详解

    锁升级的顺序:偏向锁 -> 轻量级锁 ->重量级锁 偏向锁:1、使用了synchronized关键字的代码,如果没...

  • 深入理解Java中的锁

    Java中的锁 常见的锁有synchronized、volatile、偏向锁、轻量级锁、重量级锁 1、synchr...

  • 多线程与高并发

    1. synchronized 四种状态以及锁升级过程 无锁 -> 偏向锁 -> 轻量级锁(cas) -> 重量级...

  • Sychronized的原理

    synchronized是jdk原生提供的锁,底层由偏向锁、轻量级和重量级锁来回切换实现。偏向锁并不算锁,它在对象...

  • 关于synchronized的锁升级自旋问题

    1.6之后对synchronized做了优化,包括锁消除、锁粗化、自旋和锁升级的过程:偏向锁、轻量级锁、重量级锁。...

  • JDK1.6之后 Synchronized优化

    Synchronized锁性能优化偏向锁轻量级锁升级 多线程中篇 不止一次的提到过,synchronized是Ja...

  • Java锁

    1.锁2.volitaile3.final4.轻量级锁,偏向锁,重量级锁 关于Synchronized的维基百科定...

网友评论

      本文标题:多线程篇-synchronized解析-偏向锁、轻量级锁、重量级

      本文链接:https://www.haomeiwen.com/subject/wxubahtx.html