简述Java中synchronized关键字的底层工作原理

在Java中,synchronized 关键字是一个重要的同步机制,用于控制多线程对共享资源的访问,以防止并发问题。了解 synchronized 的底层工作原理,可以帮助我们更好地编写线程安全的代码。synchronized 关键字可以应用于方法或者代码块,其底层实现依赖于Java虚拟机(JVM)中的监视器锁(Monitor Lock)或内部锁。

1. 锁的获取和释放

当线程进入一个 synchronized 方法或代码块时,它会自动获取锁。当线程离开 synchronized 方法或代码块时,无论是通过正常路径返回还是通过抛出异常,它都会自动释放锁。

2. 锁的类型

  • 轻量级锁:当没有竞争出现时,JVM会使用轻量级锁。这种锁利用CAS(比较并交换)操作,试图不暂停持有锁的线程的执行,而是通过在对象头上的标记字段中记录锁的状态来实现。
  • 重量级锁:当锁竞争激烈时,轻量级锁会升级为重量级锁。这种锁会使其他试图进入同步代码块的线程进入阻塞状态。

3. 对象头和锁状态

Java对象在内存中有一个对象头部,其中包含了锁的信息。根据锁的状态,对象头部的标记字段会有不同的设置:

  • 无锁状态:没有线程持有锁。
  • 偏向锁:一种优化手段,偏向第一个获取它的线程,进入同步块时不需要真正的竞争锁。
  • 轻量级锁:在无实际竞争的情况下使用CAS设置锁定状态。
  • 重量级锁:当有多个线程竞争同一个锁时,锁会升级到重量级锁,相关线程可能会被挂起。

4. 锁的升级过程

  • 初始状态是无锁。
  • 当第一个线程访问同步块时,锁可以升级为偏向锁。
  • 如果另一个线程尝试访问同步块,偏向锁可以脱离偏向模式并升级为轻量级锁。
  • 如果竞争加剧,轻量级锁会升级为重量级锁。

5. 监视器锁和等待/通知机制

JVM内部使用监视器锁来实现 synchronized。监视器锁包含两个队列:等待队列和阻塞队列。等待队列用于存放调用了wait()方法的线程,阻塞队列用于存放尝试获取锁但未成功的线程。当锁被释放时,来自阻塞队列的线程将有机会获取锁,而来自等待队列的线程则需要等到特定条件满足并收到notify()notifyAll()调用。

总结

synchronized 的底层实现复杂且高效,通过锁的不同状态和锁升级机制,JVM可以在运行时根据具体情况调整同步策略,从而提供良好的性能和线程安全保障。

相关推荐
岁岁种桃花儿1 分钟前
kubenetes从入门到上天系列第二十六篇:Kubernetes的Istio服务网格实战
java·kubernetes·istio
JavaEdge.3 分钟前
openclaw 本地部署实战:含网关启动 + 本地模型接入完整步骤
java·openclaw
Jelena157795857923 分钟前
1688.item_get_app接口:包装尺寸重量信息深度解析
开发语言·前端·python
loserwang4 分钟前
Fluss#1386: 从日志恢复中的 OutOfOrder 来看 LEO、HW 与 Checkpoint 的区别
java·后端
毕设源码-邱学长4 分钟前
【开题答辩全过程】以 咖啡馆管理系统的设计与实现为例,包含答辩的问题和答案
java
NGC_661110 分钟前
JDK1.7 与 JDK1.8 ConcurrentHashMap:从分段锁到桶级锁的进化
java·开发语言
大黄说说12 分钟前
PHP 数组 vs SPL 数据结构:队列与栈场景下的性能对决
开发语言·数据结构·php
rookie软工14 分钟前
Qt代理委托实现
开发语言·python·qt
leaves falling18 分钟前
C++类和对象(3)(初始化列表,类型转换,static成员,友元)
java·开发语言·c++
宵时待雨19 分钟前
C++笔记归纳15:封装map & set
开发语言·数据结构·c++·笔记·算法