【JAVA进阶篇教学】第十二篇:Java中ReentrantReadWriteLock锁讲解

博主打算从0-1讲解下java进阶篇教学,今天教学第十二篇:Java中ReentrantReadWriteLock锁讲解。 

在并发编程中,读写锁(ReadWriteLock)是一种用于管理对共享资源的访问的锁机制,它提供了比传统的互斥锁更高的并发性能。在 Java 中,读写锁是一种用于实现多线程环境下读写操作并发控制的锁机制。读写锁可以提高并发性能,因为它允许多个线程同时读取共享资源,但只允许一个线程写入共享资源。本文将详细介绍读写锁的原理,并通过代码示例说明如何保证读写锁的安全性。

目录

一、原理 

二、读写锁的使用

三、读写锁的安全性 

四、总结


一、原理 

读写锁维护了一对锁,一个读锁和一个写锁。读锁可以被多个线程同时持有,而写锁只能被一个线程持有。当一个线程获取了读锁时,其他线程可以继续获取读锁,但不能获取写锁。当一个线程获取了写锁时,其他线程既不能获取读锁也不能获取写锁。

读写锁的实现通常基于以下两个条件:

  • 读读共享:多个线程可以同时持有读锁,共享读取共享资源。
  • 读写互斥:读锁和写锁是互斥的,即一个线程获取了写锁后,其他线程不能获取读锁或写锁。
  • 写写互斥:写锁是独占的,即只有一个线程可以获取写锁。

详细说明:

  • 读锁(Read Lock):允许多个线程同时获得读锁,在没有写锁的情况下,多个线程可以并发地读取共享资源。
  • 写锁(Write Lock):独占锁,一次只允许一个线程获取写锁进行写操作,当有线程持有写锁时,其他线程无法获取读锁或写锁。
  • 读写锁实现类:Java中的ReentrantReadWriteLock是一个读写锁的实现类,它实现了ReadWriteLock接口,提供了读锁和写锁的管理

二、读写锁的使用

import java.util.concurrent.locks.ReentrantReadWriteLock;public class ReadWriteLockExample {private ReentrantReadWriteLock lock = new ReentrantReadWriteLock();public void read() {lock.readLock().lock();try {// 读取共享资源System.out.println("读取共享资源");} finally {lock.readLock().unlock();}}public void write() {lock.writeLock().lock();try {Thread.sleep(2000);// 写入共享资源System.out.println("写入共享资源");} catch (InterruptedException e) {e.printStackTrace();} finally {lock.writeLock().unlock();}}public static void main(String[] args) {ReadWriteLockExample example = new ReadWriteLockExample();// 创建并启动多个读取线程Thread thread1 = new Thread(() -> {example.read();});Thread thread2 = new Thread(() -> {example.read();});// 创建并启动写入线程Thread thread3 = new Thread(() -> {example.write();});thread1.start();thread2.start();thread3.start();// 等待线程执行完毕try {thread1.join();thread2.join();thread3.join();} catch (InterruptedException e) {e.printStackTrace();}}
}

在上述示例中,我们创建了一个ReentrantReadWriteLock对象lock,用于实现读写锁的功能。然后,我们定义了两个方法read和write,分别用于读取和写入共享资源。

  • 在read方法中,我们使用lock.readLock().lock()获取读锁,然后读取共享资源。在读取完成后,使用lock.readLock().unlock()释放读锁。
  • 在write方法中,我们使用lock.writeLock().lock()获取写锁,然后写入共享资源。在写入完成后,使用lock.writeLock().unlock()释放写锁。
  • 在main方法中,我们创建了两个读取线程thread1和thread2,以及一个写入线程thread3。然后,我们启动这些线程,并使用join方法等待它们执行完毕。

通过这种方式,我们可以使用读写锁来实现多线程环境下的读写操作并发控制,提高并发性能。

三、读写锁的安全性 

在使用读写锁时,需要注意以下几点,以保证读写锁的安全性:

  • 读写锁的公平性:ReentrantReadWriteLock类提供了公平和非公平两种实现方式。公平锁保证线程按照先来先服务的顺序获取锁,非公平锁允许线程抢占锁。在实际应用中,根据需要选择合适的公平性策略。
  • 读写锁的重入性:读写锁支持重入性,即一个线程可以多次获取同一把读写锁。在重入时,需要注意避免死锁的发生。
  • 读写锁的降级:读写锁支持降级,即一个线程可以先获取写锁,然后再获取读锁。在降级时,需要注意避免死锁的发生。
  • 读写锁的升级:读写锁不支持升级,即一个线程不能先获取读锁,然后再获取写锁。如果需要升级锁,可以先释放读锁,然后再获取写锁。
  • 读写锁的Condition:读写锁提供了条件变量的支持,可以用于实现线程的等待和通知机制。在使用条件变量时,需要注意避免死锁的发生。

四、总结

读写锁是一种用于实现多线程环境下读写操作并发控制的锁机制。读写锁可以提高并发性能,特别是在多读少写的场景下。在使用读写锁时,需要注意读写锁的公平性、重入性、降级、升级和Condition等问题,以保证读写锁的安全性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/836487.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

栈的讲解

栈的概念及结构 栈:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。 进行数据插入和删除操作的一端称为栈顶,另一端称为栈底(因为先进后出)。栈中的数据元素遵守后进先出LIFO(Last In Firs…

数据结构与算法===回溯法

文章目录 原理使用场景括号生成代码 小结 原理 回溯法是采用试错的思想,它尝试分步骤的去解决一个问题。在分步骤解决问题的过程中,当它通过尝试发现现有的分步答案不能得到有效的正确的解答的时候,它将取消上一步甚至是上几步的计算&#x…

函数模板底层本质

#include<iostream> using namespace std;template<typename T1> T1 ave(T1 a, T1 b) { return a b; }int main() {ave(100, 200);ave(short(100), short(200));return 0; }反汇编代码 模板本质是编译器帮忙生成了不同的函数 就算非类型参数值不一样编译器也重新…

HCIP(BGP综合实验)--8

一&#xff1a;实验要求 二&#xff1a;实现过程 &#xff08;一&#xff09;配置IP地址&#xff1a; AR1: [AR1]int g0/0/0 [AR1-GigabitEthernet0/0/0]ip add 12.1.1.1 24 [AR1-GigabitEthernet0/0/0]int l0 [AR1-LoopBack0]ip add 172.16.0.1 32 [AR1-LoopBack0]int l1 […

TM1650 并联在I2C 信号线的处理方法

目的是可以并联多个TM1650 在标准I2C 总线上&#xff0c;并且不影响其他标准I2C 器件。思路就是拿个额外的开关控制每一片TM1650 的使能&#xff0c;就像SPI 的CS 信号那样。 协议 TM1650 的通信协议虽说不是标准I2C&#xff0c;但也算是比较兼容的&#xff0c;比方说&#x…

springboot实现Aop

一、原理 AOP&#xff08;Aspect Oriented Programming&#xff09;的意思是&#xff1a;面向切面编程&#xff0c;通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。利用AOP可对业务逻辑进行增强&#xff0c;在不改变原有逻辑的基础上&#xff0c;在其前后进…

YOLOv8独家原创改进: AKConv(可改变核卷积)

1.AKConv原理介绍 地址:2311.11587 (arxiv.org) 摘要:基于卷积运算的神经网络在深度学习领域取得了令人瞩目的成果,但标准卷积运算存在两个固有的缺陷。一方面,卷积运算仅限于局部窗口,无法捕获其他位置的信息, 并且它的采样形状是固定的。 另一方面,卷积核的大小固定为…

基于yolov8+gradio目标检测演示系统设计

YOLOv8与Gradio&#xff1a;开启目标检测的可视化新篇章 随着人工智能技术的飞速发展&#xff0c;目标检测作为计算机视觉领域的重要分支&#xff0c;已经广泛应用于安防监控、自动驾驶、医疗影像等多个领域。而YOLO&#xff08;You Only Look Once&#xff09;系列算法作为目…

Maven多环境与SpringBoot多环境配置

1. Maven多环境配置与应用 1.1 多环境开发 我们平常都是在自己的开发环境进行开发&#xff0c; 当开发完成后&#xff0c;需要把开发的功能部署到测试环境供测试人员进行测试使用&#xff0c; 等测试人员测试通过后&#xff0c;我们会将项目部署到生成环境上线使用。 这个时…

Unity曲线插件Dreamteck Splines生成曲线Mesh

一、需求 脱离编辑器&#xff0c;运行时添加点&#xff0c;动态生成管道、线缆等曲线Mesh。 二、Dreamteck Splines简单运用 这方面资料不多&#xff0c;只有官方文档全英参考&#xff0c;而且又介绍得不详细。 2个重要组件介绍&#xff1a; SplineComputer&#xff1a; 最…

yum、rpm相关命令-工具手册

1.rpm文件: 1.1安装rpm文件: rpm -ivh kde-select.rpm23 #--nodeps强制安装,无视环境缺少依赖的检查 rpm -ivh --nodeps kde-select.rpm #--force --replacefiles可以无视rpm的冲突去强制替换(如两个rpm的安装路径相同了会有冲突) rpm -ivh --nodeps --force --replacef…

springboot中mybatisplus注意事项

使用代码生成工具CodeGenerator 需要修改的内容 dsc.setUsername(“root”); mysql账号dsc.setPassword(“root”); mysql密码strategy.setInclude(“crm_edu”); 表名pc.setModuleName(“eduservice”); //模块名 package com.test.demo;import com.baomidou.mybatisplus.a…

【超详细】跑通YOLOv8之深度学习环境配置2

环境配置2下载安装内容如下&#xff1a; CUDA&#xff1a;https://developer.nvidia.com/cuda-toolkit-archive cudnn&#xff1a;https://developer.nvidia.com/rdp/cudnn-archive 版本&#xff1a;CUDA11.3 cudnn8.9.7 CUDA安装 简介 CUDA&#xff08;Compute Unified De…

SEO之为什么研究关键词(一)

初创企业需要建站的朋友看这篇文章&#xff0c;谢谢支持&#xff1a; 我给不会敲代码又想搭建网站的人建议 新手上云 初做网站的人很容易犯的最大错误之一是&#xff0c;脑袋一拍就贸然进入某个领域&#xff0c;跳过竞争研究&#xff0c;没规划好目标关键词就开始做网站。这样做…

一个递推通项公式研究

递推关系为a(n) ​pa(n−1) ​ qa(n−2) ​&#xff0c;本项前一项*2前前项&#xff0c;具体如 1&#xff0c;1&#xff0c;3&#xff0c;7&#xff0c;17&#xff0c;41&#xff0c;99&#xff0c;239&#xff0c;…… 一般的递推关系可以用以下方法 得两个解&#xff1a; …

python算法demo0512

最长回文数 代码 class Solution:def longestPalindrome(self, s: str) -> str:n len(s)if n < 2:return smax_len 1begin 0# dp[i][j] 表示 s[i..j] 是否是回文串dp [[False] * n for _ in range(n)]for i in range(n):dp[i][i] True# 递推开始# 先枚举子串长度fo…

windows窗口消息队列与消息过程处理函数

在Windows窗口应用程序中&#xff0c;消息队列和窗口过程函数是实现消息驱动机制的核心组件。 消息队列&#xff08;Message Queue&#xff09;&#xff1a; 消息队列是用于存储窗口消息的缓冲区。当用户与应用程序交互时&#xff0c;系统会将生成的消息插入到消息队列中&…

十三、Redis哨兵模式--Sentinel

上一篇介绍了Redis中的主从复制。我们知道Redis主从中一般只有主节点对外提供写操作&#xff0c;如果主节点发生故障&#xff0c;为了保证Redis的可用性&#xff0c;这时就要在可用的slave节点中&#xff0c;挑选一个作为主节点。这种切换操作如果是人为的操作&#xff0c;那么…

支持播放h265的插件

插件源码地址&#xff1a;GitCode - 开发者的代码家园https://gitcode.com/mirrors/nanguantong/flv-h265.js/overview 1. 下载代码 运行以下命令 npm i npm run build npm run build 后生成 flv.min.js 文件&#xff0c;引入使用 2. 调用 js 文件&#xff1a; let flvj…

uniapp开发小程序使用vue的v-html解析富文本图片过大过宽显示超过屏幕解决办法

如果没有设置的话&#xff0c;就会导致图片溢出&#xff0c;过宽显示或者错位显示&#xff0c;显示效果非常的丑陋&#xff1a; 修改后显示的效果&#xff1a; 网上比较low的解决办法&#xff1a;网上各种解决方法核心思想就是在数据层把数据模板上的img数据加上style样式&…