Java并发编程:读写锁与普通互斥锁的深度对比
在Java并发编程中,锁是实现线程安全的重要工具。其中,普通互斥锁(如synchronized
和ReentrantLock
)和读写锁(ReentrantReadWriteLock
)是两种常用的同步机制。本文将从多个维度深入分析它们的区别、适用场景及性能差异,并通过示例代码展示如何在实际项目中合理选择。
一、核心概念对比
1. 普通互斥锁(Mutex)
普通互斥锁是最基本的同步机制,它遵循"排他性"原则:
- 同一时间仅允许一个线程访问共享资源,无论该线程是读操作还是写操作。
- 典型实现:
synchronized
关键字ReentrantLock
类
示例代码:
private final Lock mutex = new ReentrantLock();
private List<String> sharedList = new ArrayList<>();public void write(String data) {mutex.lock();try {sharedList.add(data);} finally {mutex.unlock();}
}public String read(int index) {mutex.lock();try {return sharedList.get(index);} finally {mutex.unlock();}
}
2. 读写锁(ReadWriteLock)
读写锁将锁分为"读锁"和"写锁",并提供更细粒度的访问控制:
- 读锁(共享锁):允许多个线程同时获取读锁,并发读取共享资源。
- 写锁(排他锁):同一时间仅允许一个线程获取写锁,且写锁存在时不允许任何线程获取读锁。
- 典型实现:
ReentrantReadWriteLock
示例代码:
private final ReadWriteLock rwLock = new ReentrantReadWriteLock();
private final Lock readLock = rwLock.readLock();
private final Lock writeLock = rwLock.writeLock();
private List<String> sharedList = new ArrayList<>();public void write(String data) {writeLock.lock();try {sharedList.add(data);} finally {writeLock.unlock();}
}public String read(int index) {readLock.lock();try {return sharedList.get(index);} finally {readLock.unlock();}
}
二、关键区别详解
1. 锁的粒度与并发度
维度 | 普通互斥锁 | 读写锁 |
---|---|---|
锁粒度 | 粗粒度(不区分读写) | 细粒度(区分读写) |
并发度 | 同一时间仅一个线程访问 | 同一时间可多个线程读或一个线程写 |
吞吐量 | 低(尤其读多写少场景) | 高(读多写少场景显著提升) |
2. 适用场景对比
场景 | 普通互斥锁 | 读写锁 |
---|---|---|
读写操作频率接近 | ✅ 简单高效 | ❌ 状态管理开销可能更高 |
读操作远多于写操作 | ❌ 吞吐量瓶颈 | ✅ 并发读性能显著提升 |
写操作占主导 | ✅ 实现简单 | ❌ 需处理写锁饥饿问题 |
需保证强一致性 | ✅ 读写均互斥 | ❌ 写锁释放前可能有读线程 |
3. 饥饿问题
- 普通互斥锁:公平模式下较少出现饥饿,但非公平模式可能导致某些线程长时间无法获取锁。
- 读写锁:默认非公平模式下,写锁可能因读锁持续被获取而长时间等待(写锁饥饿)。
解决方案:
// 创建公平读写锁,按请求顺序分配锁
private final ReadWriteLock rwLock = new ReentrantReadWriteLock(true);
三、性能对比测试
1. 测试环境
- 硬件:Intel i7-8700K CPU @ 3.70GHz,16GB RAM
- JDK:Java 17
- 测试工具:JMH
- 测试场景:模拟100线程并发访问,读:写比例分别为9:1、5:5、1:9
2. 测试结果
读:写比例 | 普通互斥锁吞吐量(ops/sec) | 读写锁吞吐量(ops/sec) | 性能提升 |
---|---|---|---|
9:1 | 54,231 | 187,629 | ~246% |
5:5 | 82,145 | 95,312 | ~16% |
1:9 | 78,321 | 62,419 | -20% |
3. 结果分析
- 读多写少场景:读写锁通过允许多线程并发读,显著提升吞吐量。
- 读写均衡场景:读写锁的性能优势减弱,因其状态管理开销高于普通互斥锁。
- 写多场景:读写锁的性能甚至低于普通互斥锁,因此时写锁的排他性导致锁竞争加剧。
四、读写锁的进阶特性
1. 锁降级(Write→Read)
写锁可降级为读锁,保证数据可见性:
public void upgradeExample() {writeLock.lock();try {// 写操作...// 降级为读锁readLock.lock();try {// 释放写锁,但仍持有读锁writeLock.unlock();// 执行读操作...} finally {readLock.unlock();}} finally {if (writeLock.isHeldByCurrentThread()) {writeLock.unlock();}}
}
2. 锁升级(Read→Write)
不推荐直接升级读锁为写锁,可能导致死锁:
public void wrongUpgrade() {readLock.lock();try {// 错误示例:不可直接升级读锁为写锁// 会导致死锁(需先释放读锁)writeLock.lock(); try {// ...} finally {writeLock.unlock();}} finally {readLock.unlock();}
}
五、最佳实践建议
1. 选择策略
- 优先考虑读写锁:当读操作占比超过70%时,读写锁通常能带来显著性能提升。
- 谨慎使用公平模式:公平模式会降低吞吐量,仅在需严格避免饥饿时使用。
- 避免锁升级:如需同时读写,建议先获取写锁,再降级为读锁。
2. 性能优化
- 分段锁:对大型数据结构分区加锁(如
ConcurrentHashMap
的实现)。 - 读写分离:将读操作和写操作分发到不同的服务实例。
- 异步写回:对写操作性能敏感的场景,可将写操作异步化(如写入队列后立即返回)。
六、总结
普通互斥锁和读写锁各有其适用场景,合理选择能显著提升系统性能:
场景 | 推荐锁类型 | 关键理由 |
---|---|---|
缓存系统(读多写少) | ReentrantReadWriteLock | 并发读性能提升明显 |
计数器更新(写操作频繁) | ReentrantLock | 读写锁状态管理开销反而降低性能 |
强一致性要求的金融系统 | synchronized/ReentrantLock | 避免读写锁的并发读带来的一致性问题 |
配置中心(读操作占绝对主导) | StampedLock(乐观读) | 进一步提升无竞争读的性能 |
在实际开发中,建议通过JMH等工具进行性能基准测试,验证锁选择的合理性。同时,注意监控锁竞争情况(如通过JVM工具查看锁等待时间),及时调整锁策略。
相关文章:
Java并发编程:读写锁与普通互斥锁的深度对比
在Java并发编程中,锁是实现线程安全的重要工具。其中,普通互斥锁(如synchronized和ReentrantLock)和读写锁(ReentrantReadWriteLock)是两种常用的同步机制。本文将从多个维度深入分析它们的区别、适用场景及…...
Spring Boot Actuator未授权访问漏洞修复
方案1:在网关的配置文件里增加以下配置 management:endpoints:web:exposure:include: []enabled-by-default: falseendpoint:health:show-details: ALWAYS 方案二:直接在nginx配置拦截actuator相关接口 location /actuator { return 403; …...

机器学习——SVM
1.什么是SVM 支持向量机(support vector machines,SVM)是一种二分类模型,它将实例的特征向量映射为空间中的一些点,SVM 的目的就是想要画出一条线,以 “最好地” 区分这两类点,以至如果以后有了…...

【音视频】FFmpeg 硬件(NVDIA)编码H264
FFmpeg 与x264的关系 ffmpeg软编码是使⽤x264开源项⽬,也就是说ffmpeg软编码H264最终是调⽤了x264开源项⽬,所以我们要先理解ffmpeg和x264的调⽤关系,这⾥我们主要关注x264_init。对于x264的参数都在 ffmpeg\libavcodec \libx264.c x264\co…...

贪心算法应用:超图匹配问题详解
贪心算法应用:超图匹配问题详解 贪心算法在超图匹配问题中有着广泛的应用。下面我将从基础概念到具体实现,全面详细地讲解超图匹配问题及其贪心算法解决方案。 一、超图匹配问题基础 1. 超图基本概念 **超图(Hypergraph)**是普…...
OpenCV CUDA模块结构分析与形状描述符------计算指定阶数的矩(Moments)所需的总数量函数:numMoments
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 该函数用于计算指定阶数的矩(Moments)所需的总数量。 在图像处理中,矩(moments)是一…...

【Web应用】若依框架:基础篇13 源码阅读-前端代码分析
文章目录 ⭐前言⭐一、课程讲解过程⭐二、自己动手实操⭐总结 标题详情作者JosieBook头衔CSDN博客专家资格、阿里云社区专家博主、软件设计工程师博客内容开源、框架、软件工程、全栈(,NET/Java/Python/C)、数据库、操作系统、大数据、人工智能、工控、网…...

[java八股文][JavaSpring面试篇]SpringCloud
了解SpringCloud吗,说一下他和SpringBoot的区别 Spring Boot是用于构建单个Spring应用的框架,而Spring Cloud则是用于构建分布式系统中的微服务架构的工具,Spring Cloud提供了服务注册与发现、负载均衡、断路器、网关等功能。 两者可以结合…...
深度学习篇---face-recognition的优劣点
face_recognition库是一个基于 Python 的开源人脸识别工具,封装了 dlib 库的深度学习模型,具有易用性高、集成度强的特点。以下从技术实现、应用场景等维度分析其优劣势: 一、核心优势 1. 极简 API 设计,开发效率极高 代码量少:几行代码即可实现人脸检测、特征提取和比对…...

基于分布式状态机的集装箱智能道口软件架构方法
集装箱码头对进出场道口的通过能力始终是要求最高的,衡量道口的直接指标为道口通行效率,道口通行效率直接体现了集装箱码头的作业效率以及对外服务水平,进而直接影响到码头的综合能力。所以,码头普遍使用智能道口实现24小时无人值…...
Oracle的Hint
racle的Hint是用来提示Oracle的优化器,用来选择用户期望的执行计划。在许多情况下,Oracle默认的执行方式并不总是最优的,只不过由于平时操作的数据量比较小,所以,好的执行计划与差的执行计划所消耗的时间差异不大&…...
手动事务的使用
使用原因: 公司需要写一个定时任务,涉及增改查操作, 定时将前端页面配置的字典数据(标签数据)同步到数据库特定的表(标签表) 查询字典表数据 字典有,数据库表没有新增 都有,判断名称,名称不同修…...

Vue 树状结构控件
1、效果图如下所示: 2、网络请求的数据结构如下: 3、新建插件文件:menu-tree.vue,插件代码如下: <template><div class"root"><div class"parent" click"onParentClick(pare…...
Spring Boot的启动流程,以及各个扩展点的执行顺序
目录 1. 初始化阶段执行顺序 1.1 Bean的构造方法(构造函数) 1.2 PostConstruct 注解方法 1.3 InitializingBean 的 afterPropertiesSet() 1.4 Bean(initMethod "自定义方法") 2. 上下文就绪后的扩展点 2.1 ApplicationContext 事件监听…...

【LUT技术专题】图像自适应3DLUT代码讲解
本文是对图像自适应3DLUT技术的代码解读,原文解读请看图像自适应3DLUT文章讲解 1、原文概要 结合3D LUT和CNN,使用成对和非成对的数据集进行训练,训练后能够完成自动的图像增强,同时还可以做到极低的资源消耗。下图为整个模型的…...
Apache Doris 在数据仓库中的作用与应用实践
在当今数字化时代,企业数据呈爆炸式增长,数据仓库作为企业数据管理和分析的核心基础设施,其重要性不言而喻。而 Apache Doris,作为一款基于 MPP(Massively Parallel Processing,大规模并行处理)…...

vscode使用“EIDE”和“Cortex-Debug”插件利用st-link插件实现程序烧写以及调试工作
第一步:安装vscode插件“EIDE”EIDE和“Cortex-Debug”。 第二步:配置EIDE 2.1安装“实用工具”: 2.2 EIDE插件配置:根据安装的keil C51 keil MDK IAR的相关路径设置 第三步:配置Cortex-Debug插件 点击settings.jso…...

Spring @Value注解的依赖注入实现原理
Spring Value注解的依赖注入实现原理 一,什么是Value注解的依赖注入二,实现原理三,代码实现1. 定义 Value 注解2. 实现 InstantiationAwareBeanPostProcessor3. 实现 AutowiredAnnotationBeanPostProcessor4. 占位符解析逻辑5. 定义 StringVa…...

三、kafka消费的全流程
五、多线程安全问题 1、多线程安全的定义 使用多线程访问一个资源,这个资源始终都能表现出正确的行为。 不被运行的环境影响、多线程可以交替访问、不需要任何额外的同步和协同。 2、Java实现多线程安全生产者 这里只是模拟多线程环境下使用生产者发送消息&…...
商品模块中的多规格设计:实现方式与电商/ERP系统的架构对比
在商品管理系统中,多规格设计(Multi-Specification Product Design)是一个至关重要但又极具挑战性的领域。无论是面向消费者的电商系统,还是面向企业管理的ERP系统,对商品规格的处理方式直接影响库存管理、订单履约、数…...
(三)动手学线性神经网络:从数学原理到代码实现
1 线性回归 线性回归是一种基本的预测模型,用于根据输入特征预测连续的输出值。它是机器学习和深度学习中最简单的模型之一,但却是理解更复杂模型的基础。 1.1 线性回归的基本元素 概念理解: 线性回归假设输入特征和输出之间存在线性关系。…...

Axure形状类组件图标库(共8套)
点击下载《月下倚楼图标库(形状组件)》 原型效果:https://axhub.im/ax9/02043f78e1b4386f/#g1 摘要 本图标库集锦精心汇集了8套专为Axure设计的形状类图标资源,旨在为产品经理、UI/UX设计师以及开发人员提供丰富多样的设计素材,提升原型设计…...

20250530-C#知识:String与StringBuilder
String与StringBuilder string字符串在开发中经常被用到,不过在需要频繁对字符串进行增加和删除时,使用StringBuilder有利于提升效率。 1、String string是一种引用类型而非值类型(某些方面像值类型)使用“”进行两个string对象的…...

从 Docker 到 Containerd:Kubernetes 容器运行时迁移实战指南
一、背景 Kubernetes 自 v1.24 起移除了 dockershim,不再原生支持 Docker Engine,用户需迁移至受支持的 CRI 兼容运行时,如: Containerd(推荐,高性能、轻量级) CRI-O(专为 Kuberne…...

uniapp中view标签使用范围
不止用于微信小程序。兼容型号,是uniapp内置组件之一,在uniapp中进行了跨平台适配。支持所有uniapp的平台。如微信小程序、h5、app、支付宝小程序...
Celery 核心概念详解及示例
Celery 核心概念详解及示例 Celery 是一个简单、灵活且可靠的分布式系统,用于处理大量消息,提供对任务队列的操作,并支持任务的调度和异步执行。它常用于深度优化 Web 应用的性能和响应速度,通过将耗时的操作移到后台异步执行&am…...

欢乐熊大话蓝牙知识14:用 STM32 或 EFR32 实现 BLE 通信模块:从0到蓝牙,你也能搞!
🚀 用 STM32 或 EFR32 实现 BLE 通信模块:从0到蓝牙,你也能搞! “我能不能自己用 STM32 或 EFR32 实现一个 BLE 模块?” 答案当然是:能!还能很帅! 👨🏭 前…...

IDEA 在公司内网配置gitlab
赋值项目链接 HTTPS 将HTTP的链接 ip地址换成 内网地址 例如:https:172.16.100.18/...... 如果出现需要需要Token验证的情况: 参考:Idea2024中拉取代码时GitLab提示输入token的问题_gitlab token-CSDN博客...

黑马Java面试笔记之 微服务篇(业务)
一. 限流 你们项目中有没有做过限流?怎么做的? 为什么要限流呢? 一是并发的确大(突发流量) 二是防止用户恶意刷接口 限流的实现方式: Tomcat:可以设置最大连接数 可以通过maxThreads设置最大Tomcat连接数,实现限流,但是适用于单体架构 Nginx:漏桶算法网关,令牌桶算法自定…...

通过WiFi无线连接小米手机摄像头到电脑的方法
通过WiFi无线连接小米手机摄像头到电脑的方法 以下是基于Scrcpy和DroidCam两种工具的无线连接方案,需提前完成开发者模式与USB调试的开启(参考原教程步骤): 方法一:Scrcpy无线投屏(无需手机端安装…...