Java并发编程之ReadWriteLock读写锁的操作方法
作者:Java硬件工程师 发布时间:2023-12-07 20:08:37
1.ReadWriteLock介绍
为什么我们有了Lock,还要用ReadWriteLock呢。我们对共享资源加锁之后,所有的线程都将会等待。Lock读操作也锁,写操作也会锁,而对共享资源读的时候,其实是不用加锁的。当然读写同时存在的情况也会有。
比如我们数据库常用操作有增删改查,增删改都是写操作,写操作必须加锁,而读操作可以共享。不是所有的操作都需要加锁。
为了进一步提高复用性和粒度,写操作独占,读操作共享,不加锁。
ReadWriteLock管理一组锁,一个是只读的锁,一个是写锁。读锁可以在没有写锁的时候被多个线程同时持有,写锁是独占的。
即读读可共享,写读写写要独占
读操作不用关心资源争抢和数据一致性的操作
多个线程同时读一个资源类没有任何问题,所以为了满足并发量,读取共享资源应该可以同时进行。
但是如果有一个线程想去写共享资源来,就不应该再有其它线程可以对该资源进行读或写。
小总结:
读-读能共存
读-写不能共存
写-写不能共存
ReadWriteLock适用于读多写少的并 * 况。
Java并发包中ReadWriteLock是一个接口,主要有两个方法,如下:
public interface ReadWriteLock {
/**
* 返回读锁
*/
Lock readLock();
/**
* 返回写锁
*/
Lock writeLock();
}
Java并发库中ReetrantReadWriteLock实现了ReadWriteLock接口并添加了可重入的特性。
2.不用读写锁的案例
class MyCache{
//volalite保证数据再线程间的可见性和一定的有序性
private volatile Map<String,Object> map=new HashMap<>();
public void put(String key,Object value){
System.out.println(Thread.currentThread().getName()+"\t ---写入数据"+key);
map.put(key,value);
try {TimeUnit.SECONDS.sleep(3);} catch (InterruptedException e) {e.printStackTrace();}
System.out.println(Thread.currentThread().getName()+"\t ---写入完成");
}
public void get(String key){
System.out.println(Thread.currentThread().getName()+"\t 读取数据");
Object result=map.get(key);
System.out.println(Thread.currentThread().getName()+"\t 读取完成"+result);
}
}
public class ReadWriteLockDemo {
public static void main(String[] args) {
// TODO Auto-generated method stub
MyCache myCache=new MyCache();
for(int i=1;i<=5;i++){
final int tempInt=i;
new Thread(()->{
myCache.put(tempInt+"",tempInt+"");
},String.valueOf(i)).start();
}
for(int i=1;i<=5;i++){
final int tempInt=i;
new Thread(()->{
myCache.get(tempInt+"");
},String.valueOf(i)).start();
}
}
}
上面的运行结果出现了一个很严重的问题,比如说,线程1在写入数据的时候,还没等线程1写完,其它线程也在写和读。这违背了事务的原子性。而且也破坏了数据的完整性和一致性,我正在写的时候,不允许其他线程同时也在写或者读。即1写入的时候,必须要满足后一条就是1写入完成。
3.用读写锁的案例
class MyCache{
//volalite保证数据再线程间的可见性和一定的有序性
private volatile Map<String,Object> map=new HashMap<>();
private ReadWriteLock readwritelock=new ReentrantReadWriteLock();
public void put(String key,Object value){
readwritelock.writeLock().lock();
try {
System.out.println(Thread.currentThread().getName()+"\t ---写入数据"+key);
map.put(key,value);
try {TimeUnit.SECONDS.sleep(3);} catch (InterruptedException e) {e.printStackTrace();}
System.out.println(Thread.currentThread().getName()+"\t ---写入完成");
} catch (Exception e) {
// TODO: handle exception
e.printStackTrace();
}finally{
readwritelock.writeLock().unlock();
}
}
public void get(String key){
readwritelock.readLock().lock();
try {
System.out.println(Thread.currentThread().getName()+"\t 读取数据");
Object result=map.get(key);
System.out.println(Thread.currentThread().getName()+"\t 读取完成"+result);
} catch (Exception e) {
// TODO: handle exception
e.printStackTrace();
}finally{
readwritelock.readLock().unlock();
}
}
}
public class ReadWriteLockDemo {
public static void main(String[] args) {
// TODO Auto-generated method stub
MyCache myCache=new MyCache();
for(int i=1;i<=5;i++){
final int tempInt=i;
new Thread(()->{
myCache.put(tempInt+"",tempInt+"");
},String.valueOf(i)).start();
}
for(int i=1;i<=5;i++){
final int tempInt=i;
new Thread(()->{
myCache.get(tempInt+"");
},String.valueOf(i)).start();
}
}
}
来源:https://blog.csdn.net/qq_39736597/article/details/113331843


猜你喜欢
- using System;using System.Collections.Generic;using System.Text;using
- 本教程源码https://github.com/bestaone/HiAuth源码比较全面,教程我就只介绍关键代码了,喜欢的点个star,谢
- 前言这几天琢磨着开发个个人作品的时候,发现原来Unity3D官方没有提供圆锥体的创建功能,就自己做了个编辑器扩展。鉴于之前搜索Mesh编程的
- 用Stream解决两层List属性求和假设一个人有很多个银行账户,每个银行账户中存有不同金额的存款,那么我们如何用Stream求一组人的所有
- 1.Bean 的创建生命周期UserService.class —> 无参构造方法(推断构造方法) &md
- 示例【通过班级查询老师信息】创建t_classes创建t_classessTeacher创建t_teacher创建Classespackag
- /* 系统名:SaleManage* 模块名:SortPags* 模块说明:排序分页类(传入DataTable,及相关信息,然后分页,并排序
- 什么是JPA一种规范,并非ORM框架,也就是ORM上统一的规范spring-boot-starter-data-jpa 
- 目录安装 BenchmarkDotNet什么是基准测试创建基准测试代码运行 benchmarkBenchmarkDotNet 是一个轻量级,
- 前言上篇博客我们介绍了如何创建一个spring项目,并且如何的存、取对象,介绍了相关方法,那么本篇博客将接着上篇博客的内容介绍如何更加简单的
- 昨天看了一段android配置aspectj实现AOP的直播视频,就试着自己配置了一下,可能是因为我自己的AndroidStudio环境的问
- 本文实例为大家分享了Android实现仪表盘效果的具体代码,供大家参考,具体内容如下仪表盘效果,圆弧可变色,效果图如下:通过自定义view实
- 在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体
- 1、概述首先和大家一起回顾一下Java 消息服务,在我之前的博客《Java消息队列-JMS概述》中,我为大家分析了:1.消息服务:一个中间件
- 在数字科技日新月异的今天,软件和硬件的完美结合,造就了智能移动设备
- 平时写项目的时候,java之父叫我们多打日志,我们通常使用traceId和requestId来保存完整请求的链路日志,例如市面上的skywa
- 本文实例讲述了java图片滑动验证(登录验证)原理与实现方法。分享给大家供大家参考,具体如下:这是我简单做出的效果图,处理300X150px
- 1.根据入参带分页参数进行sql查询分页 Criteria criteria = n
- volatile关键字相信了解Java多线程的读者都很清楚它的作用。volatile关键字用于声明简单类型变量,如int、float、boo
- springboottest测试依赖和使用<dependency> <groupId>or