学了这么久的多线程,你知道synchronized到底是怎么工作的吗?

synchronized原理
一. 特性
二. 加锁过程(锁升级/锁膨胀)
1. 无锁状态
2. 偏向锁
3. 轻量级锁
4. 重量级锁
5. 总结
三. 锁优化
1. 锁消除
2. 锁粗化
一. 特性
这部分内容在上篇文章中的 synchronized充当了哪些锁部分已经介绍过了哦,没有看的小伙伴可以去看看synchronized的特性

二. 加锁过程(锁升级/锁膨胀)
在Java中JVM虚拟机将synchronized锁分为无锁、偏向锁、轻量级锁、重量级锁状态。会根据不同的情况,进行不同的升级操作


1. 无锁状态
此状态理解起来较为简单,没有进行线程任务时最开始的状态就是无锁状态。

2. 偏向锁
偏向锁类似于一种乐观锁,当一个线程在执行任务时,偏向锁会给这个线程设定一个标记(并不是真正地加锁),如果后续没有其他线程来竞争这个锁,那么这个偏向锁就不会再进行其他的任何操作了,有效避免了因为加锁过程而产生的内存开销问题
若有其他线程也竞争这把锁,那么此时第一个线程会立马把锁拿到(因为之前第一个线程已经有了偏向锁标记,所以很容易拿到)然后进入轻量级锁的状态
偏向锁的大体思路是能不加锁就尽量不加锁避免内存开销,只做上标记即可,但如果实在要加锁,也会因为标记的存在而立马把锁拿到(类似于高考填志愿保底心态)

3. 轻量级锁
当进入轻量级锁锁状态(自适应自旋锁)后,是完全在用户态上实现的,且是基于CAS来完成的操作,因为这个状态不涉及到内核态和用户态的切换,也不涉及到线程的阻塞和调度过程。所以并不会对系统的内存有着过于高的开销,因此可以保证更高效地获取到锁(一个线程释放锁后,另一个线程会马上获取到锁)
具体步骤

通过 CAS 检查并更新一块内存 (比如 null => 该线程引用)
如果更新成功, 则认为加锁成功
如果更新失败, 则认为锁被占用, 继续自旋式的等待(并不放弃 CPU)
由于自旋操作可能会一直让CPU 空转,比较浪费 CPU 资源,因此此处的自旋不会一直持续进行,而是达到一定的时间(重试)次数,就不再自旋了,也就是所谓的 “自适应”(根据情况来)
4. 重量级锁
当锁的竞争变得非常激烈时,如果再按照之前自旋的方式,那么对于CPU的开销是非常高的,而且此时自旋还不能快速地获取到锁的状态,那么此时就会变成重量级锁(挂起等待锁),对于挂起等待锁来说,锁的等待过程是释放CPU的过程,此时会节省CPU的开销,但付出的代价是引入了线程的阻塞和调度的开销(以CPU资源换取性能)
具体过程
此处的重量级锁就是用到了内核提供的mutex,要执行加锁操作,首先会进入内核态,在内核态判定当前的锁是否已经被占用,若该锁没有被占用,则加锁成功,切换回用户态;若该锁被占用,则加锁失败,此时线程进入锁的等待队列去挂起等待,直到锁被其他线程释放后,操作系统才会唤醒挂起等待锁的这个线程,最后这个线程才会获取到锁

5. 总结
锁升级(锁膨胀)的过程完全是synchronized内部自适应完成的,即根据不同的情况(即锁冲突的高或低状态)来升级或降级成对应的状态,不需要用户或者程序员去干预,因此使用起来会比较方便。
2.注意, synchronized在有些JVM版本上是可以同时实现降级和升级的自适应的,但在有些JVM上只能实现升级的自适应。
三. 锁优化
————————————————
版权声明:本文为CSDN博主「春风~十一载」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/Mubei1314/article/details/120795468

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值