当前位置: 首页 > news >正文

synchronized关键字的实现

Java对象结构

synchronized锁升级过程

为了优化synchronized锁的效率,在JDK6中,HotSpot虚拟机开发团队提出了锁升级的概念,包括偏向锁、轻量级锁、重量级锁等,锁升级指的就是“无锁 --> 偏向锁 --> 轻量级锁 --> 重量级锁”。

synchronized同步锁相关信息保存到锁对象的对象头里面的Mark Word中,锁升级功能主要是依赖Mark Word中锁标志位和是否偏向锁标志位来实现的。

无锁 -> 偏向锁:第一个锁获得它的线程,会在对象头(Mark Word中)记录锁偏向的线程ID。

偏向锁 -> 轻量级锁:当A线程持有偏向锁时,此时B线程通过CAS自旋的方式去竞争锁,如果竞争成功修改偏向锁的线程ID依旧是偏向锁,否则锁会升级为轻量级锁。

轻量级锁 -> 重量级锁:当A线程持有轻量级锁时,此时B线程通过CAS自旋的方式去竞争锁,如果线程B自旋一定次数后依旧无法获得锁,则锁会升级为重量级锁。

注意:可以使用-XX:UseBiasedLocking参数关闭偏向锁,此时默认进入轻量级锁。

synchronized偏向锁的撤销

偏向锁只有遇到其他线程尝试竞争偏向锁时,持有偏向锁的线程才会释放锁,线程不会主动去释放偏向锁。偏向锁的撤销,需要等待全局安全点safepoint,它会首先暂停拥有偏向锁的线程A,然后判断这个线程A,此时有两种情况:

synchronized重量级锁的实现原理

同步分为显式同步和隐式同步,显式同步,指的是有明确的monitorenter和monitorexit指令。隐式同步,指的是同步方法是由方法调用指令读取运行时常量池中方法的ACC_SYNCHRONIZED标志来隐式实现的。synchronized修饰的方法是隐式同步(使用ACC_SYNCHRONIZED标志),synchronized修饰的同步代码块是显式同步(使用monitorenter和monitorexit指令)。

public class SyncCodeBlock {public int i;public void syncTask(){//同步代码库synchronized (this){i++;}}
}//===========主要看看syncTask方法实现================
public void syncTask();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=3, locals=3, args_size=1
0: aload_0
1: dup
2: astore_1
3: monitorenter  //注意此处,进入同步方法
4: aload_0
5: dup
6: getfield      #2             // Field i:I
9: iconst_1
10: iadd
11: putfield      #2            // Field i:I
14: aload_1
15: monitorexit   //注意此处,退出同步方法
16: goto          24
19: astore_2
20: aload_1
21: monitorexit //注意此处,退出同步方法
22: aload_2
23: athrow
24: return
Exception table:
public class SyncMethod {public int i;public synchronized void syncTask(){i++;}
}//==================syncTask方法======================
public synchronized void syncTask();
descriptor: ()V
//方法标识ACC_PUBLIC代表public修饰,ACC_SYNCHRONIZED指明该方法为同步方法
flags: ACC_PUBLIC, ACC_SYNCHRONIZED
Code:stack=3, locals=1, args_size=10: aload_01: dup2: getfield      #2                  // Field i:I5: iconst_16: iadd7: putfield      #2                  // Field i:I10: returnLineNumberTable:line 12: 0line 13: 10

显式同步是利用monitorenter和monitorexit这两个字节码指令来区分是否同步,它们分别位于同步代码块的开始和结束位置。monitorenter指令后会插入LoadLoadLoadStore 屏障,monitorexit指令后会插入StoreStoreStoreLoad 屏障。

隐式同步是利用ACC_SYNCHRONIZED标志来区分是否同步,进入方法前会插入LoadLoadLoadStore 屏障,退出方式时后会StoreStoreStoreLoad 屏障。执行线程会先获取monitor,获取成功之后才能执行方法体,方法执行完后再释放monitor。

注意:两种同步方式本质上没有区别,都是通过monitor对象来实现线程同步。

加锁流程

加锁成功

当JVM执行到monitorenter指令时,当前线程试图获取monitor对象的_Owner是否是当前线程,如果已经被当前线程所持有,_count+1,_Owner指向持有monitor对象的线程。

加锁失败

如果获取monitor对象失败,该线程则会进入阻塞队列(_EntryList),并一直阻塞。

释放锁流程

当JVM执行到monitorexit指令时,_count-1,当锁计数器为0时,_Owner恢复为null,表示该锁被释放了,如果_EntryList非空,那么通知_EntryList中所有阻塞的线程,去竞争然后成为下一个_Owner。

wait流程

当持有锁的线程调用wait()方法时,持有锁的线程会进入等待队列(_WaitSet)等待被唤醒,同时count-1,_Owner恢复为null。

notify/notifyAll流程

当有线程执行notify()/notifyAll()方法时,会随机唤醒一个或所有线程从_WaitSet队列进入到_EntryList队列中,只有线程持有者才能调用notify(),当锁释放时会通知_EntryList中所有阻塞的线程。

相关文章:

  • 第二篇:Django配置及ORM操作
  • 金融软件测试有哪些注意事项?专业第三方软件测试服务机构分享
  • 【leetcode100】组合总和Ⅳ
  • AIP代码生成器——标准化接口开发智能工具
  • 算法之分支定界
  • 多语言虚拟币海外游戏娱乐平台源码详解(整合篇)
  • 架构-系统可靠性分析与设计
  • 复杂性决策-思维训练
  • JAVA设计模式——(五)享元模式(Flyweight Pattern)
  • 缓存与数据库数据一致性:旁路缓存、读写穿透和异步写入模式解析
  • ArrayList与顺序表详解
  • C# 综合示例 库存管理系统4 classMod类
  • 力扣面试150题--基本计算器
  • 移动零--LeetCode
  • 切割PDF使用python,库PyPDF2
  • 区块链技术:深入共识算法、智能合约与DApps的架构奥秘
  • 【GIT】github中的仓库如何删除?
  • Langchain+RAG+向量数据库
  • vue-study(1)
  • java面向对象编程【基础篇】之基础语法
  • 海上生明月,九天揽星河,2025年“中国航天日”主场活动在上海启动
  • 广东东莞调整普通住宅价格标准:一类镇街上浮300余元/平方米
  • 著名茶叶专家谢丰镐逝世,享年95岁
  • 东阿至聊城公交票价取消八折优惠:运行成本高昂
  • 汪东进卸任中海油董事长,深耕油气领域40余年、已临近退休
  • 被电诈100万元又要被骗71万元,女子经民警近8小时劝阻幡然醒悟