Java减少锁竞争提升性能技巧解析
时间:2026-02-09 23:53:35 450浏览 收藏
学习文章要努力,但是不要急!今天的这篇文章《Java如何减少锁竞争提升性能解析》将会介绍到等等知识点,如果你想深入学习文章,可以关注我!我会持续更新相关文章的,希望对大家都能有所帮助!
应优先用无锁数据结构替代加锁,如ConcurrentHashMap、LongAdder、AtomicInteger等;细粒度控制需按业务拆分锁或哈希分段;避免锁升级,确保锁对象私有、final、稳定;读多写少用读写锁,极简读场景可选StampedLock。

用 ReentrantLock 替代 synchronized 做细粒度控制
synchronized 简单但粗放,锁住整个方法或代码块,容易让不相关的线程互相阻塞。换成 ReentrantLock 后,你可以按业务逻辑拆分临界区,比如把“查库存”和“扣库存”拆成两个独立锁,或对不同商品 ID 使用不同锁实例。
常见错误是直接用一个全局 ReentrantLock 代替所有 synchronized,结果锁竞争一点没少。真正有效的做法是:
- 为高频并发资源(如缓存桶、分段 Map)创建多个
ReentrantLock实例,用哈希取模分配 - 避免在
lock()和unlock()之间做耗时操作(如远程调用、IO) - 务必在
finally块中调用unlock(),否则可能永久死锁
优先用无锁数据结构替代加锁容器
不是所有共享状态都需要锁。JDK 提供的 ConcurrentHashMap、AtomicInteger、LongAdder 等,底层基于 CAS 和分段思想,在多数场景下比手动加锁快得多,且不易出错。
例如高并发计数场景,用 LongAdder 替代 synchronized + int,性能提升常达 5–10 倍;而 ConcurrentHashMap 的 computeIfAbsent 能安全完成“检查-创建-返回”三步,无需额外同步。
注意点:
ConcurrentHashMap的size()不是 O(1),它要遍历所有 segment,高并发下慎用AtomicReference在需要原子更新对象引用时比锁更轻量,但不适用于复合逻辑(如先读再条件写)- 别为了“无锁”强行套用
Unsafe或自旋锁,JVM 对synchronized的锁消除和偏向优化已很成熟
避免锁升级和 monitor 争抢:减少 synchronized 误用
JVM 对 synchronized 做了大量优化,但前提是锁对象稳定、竞争低。一旦出现锁膨胀(从偏向锁 → 轻量级锁 → 重量级锁),每次进入临界区都要触发 OS mutex,性能断崖下跌。
典型诱因包括:
- 用可变对象(如
new Object()或临时字符串)作锁,导致无法启用偏向锁 - 在循环内反复
synchronized同一对象,但每次只做微小操作,放大上下文切换开销 - 锁对象被多个无关模块共用(比如全系统都用
MyClass.class当锁)
实操建议:
- 锁对象声明为
private final Object lock = new Object();,确保唯一性和不可变性 - 把多次小同步合并为一次大同步(但需权衡持有时间,防止阻塞太久)
- 用 JFR 或
jstack观察线程堆栈里是否频繁出现Object.wait()或Unsafe.park(),这是重量级锁已生效的信号
用读写锁分离读多写少场景
如果某个状态被大量读取、极少修改(如配置中心的本地副本、路由表),ReentrantReadWriteLock 能显著提升吞吐。读锁可重入、允许多个线程同时持有;写锁独占,且会阻塞新读锁获取。
但要注意:
- 写锁获取前,所有等待的读锁和写锁都会排队,写操作频繁时反而比普通锁更慢
StampedLock更轻量,支持乐观读(tryOptimisticRead),适合读远多于写的极简场景,但它不支持重入,且使用不当易出错- 不要在读锁内调用可能升级为写锁的方法(如某些集合的
computeIfAbsent内部会写),会导致死锁
锁竞争不是靠“换更高级的锁”解决的,而是靠识别真正共享的变量、缩小临界区、以及用合适语义的数据结构替代裸锁。很多所谓“性能瓶颈”,其实只是锁范围划错了。
好了,本文到此结束,带大家了解了《Java减少锁竞争提升性能技巧解析》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
291 收藏
-
408 收藏
-
247 收藏
-
264 收藏
-
366 收藏
-
299 收藏
-
409 收藏
-
256 收藏
-
481 收藏
-
437 收藏
-
311 收藏
-
327 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习