Java多线程中有很多的锁机制,他们都有各自的应用场景,例如今天我说的这种锁机制:读写锁

读写锁,见名知意,主要可以进行两种操作,读和写操作,他们之间结合使用起来又是各不相同的。比如多个线程之间可以同时读,但不可以同时写,也不可以一边读一边写,有点类似于数据库中的共享锁以及排它锁,下面我具体事例演示:

需要注意的是,不管是进行读操作还是写操作,一定要成对去调用方法,就是开启锁后一定要关闭,且为了保险起见,关闭最好要写在finally语句块中去。
另外,我们在声明对象时,有两种方式(有参数和无参数),若为true则为公平机制,默认为false,不写也为false

简单举例演示

package day_12_27;

import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock; /**
* @author soberw
* @Classname ReadWriteLockTest
* @Description ReadWriteLock的测试
* @Date 2021-12-27 10:10
*/
public class ReadWriteLockTest {
public static void main(String[] args) {
var rwl = new RWL();
//情况一:
new Thread(() -> rwl.write(100), "A").start();
//情况二:
// new Thread(() -> rwl.write(99), "AA").start();
// new Thread(() -> rwl.write(88), "AAA").start(); for (int i = 0; i < 20; i++) {
new Thread(rwl::read, "B" + i).start();
}
}
} class RWL {
//可以传参,若为true则为公平锁,默认为false,不写也为false
private ReadWriteLock rwl = new ReentrantReadWriteLock();
public int number; void write(int number) {
rwl.writeLock().lock();
try {
this.number = number;
Thread.sleep(2000);
System.out.println(Thread.currentThread().getName() + ":start--" + number);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
rwl.writeLock().unlock();
}
} void read() {
rwl.readLock().lock();
try {
Thread.sleep(2000);
System.out.println(Thread.currentThread().getName() + ":" + number);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
rwl.readLock().unlock();
}
}
}

先分析代码,写线程两秒后写入一个线程“A”,然后开启20个读线程都去读数据,结果:

在写的时候,会停顿两秒,因为我加了sleep(),但是我在读的时候也加了阻塞,却发现20个进程还是同时执行了,这就说明多个进程之间可以同时读,且读写不能同时进行,当线程写的时候,其他所有的读线程都要等着

下面打开情况二,多个线程都进行写操作:

发现会一个一个执行,间隔两秒,并不会像读操作一样,同时进行,这说明写不能同时进行

应用场景举例

那么在实际开发中,常用这种锁机制来完成缓存系统的操作。所谓缓存系统,举个例子,就比如用户要读取某个数据的时候,我不直接去向数据库请求,而是先看我内部有没有这个数据,如果有就直接返回,如果没有,就从数据库中查找这个数,查到后将这个数据存入我内部存储器中,下次再有人来要这个数据,我就直接返回这个数不用再到数据库中找了,这就大大的避免了频繁的访问数据库。下面我就书写代码模拟这一过程:

package day_12_27;

import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock; /**
* @author soberw
* @Classname ExerData
* @Description 模拟缓存系统
* @Date 2021-12-27 20:02
*/
public class ExerData {
private static ReadWriteLock rwLock = new ReentrantReadWriteLock(); private static Map<String, Object> map = new HashMap<String, Object>(); public static void main(String[] args) {
for (int i = 0; i < 3; i++) {
final String key = i + "";
new Thread(new Runnable() { @Override
public void run() {
for (int i1 = 0; i1 < 5; i1++) { System.out.println(Thread.currentThread().getName() + " read data: " + getData(key)); }
}
}).start();
}
} public static Object getData(String key) {
rwLock.readLock().lock();
Object value = null;
try {
value = map.get(key);
if (value == null) {
rwLock.readLock().unlock();
rwLock.writeLock().lock(); try {
if (value == null) {
// 此处在实际应用中,就是从数据库中查找数据,放入缓存中
value = Thread.currentThread().getName() + new Date();
} } finally {
rwLock.writeLock().unlock();
}
rwLock.readLock().lock();
} } finally {
rwLock.readLock().unlock();
} return value;
} }

运行结果:

Java多线程之读写锁机制的更多相关文章

  1. java多线程 -- ReadWriteLock 读写锁

    写一条线程,读多条线程能够提升效率. 写写/读写 需要“互斥”;读读 不需要互斥. ReadWriteLock 维护了一对相关的锁,一个用于只读操作,另一个用于写入操作.只要没有 writer,读取锁 ...

  2. java 多线程 day12 读写锁

    import java.util.Random;import java.util.concurrent.locks.ReadWriteLock;import java.util.concurrent. ...

  3. 22、Java并发性和多线程-Java中的读/写锁

    以下内容转自http://ifeve.com/read-write-locks/: 相比Java中的锁(Locks in Java)里Lock实现,读写锁更复杂一些.假设你的程序中涉及到对一些共享资源 ...

  4. 技术笔记:Delphi多线程应用读写锁

    在多线程应用中锁是一个很简单又很复杂的技术,之所以要用到锁是因为在多进程/线程环境下,一段代码可能会被同时访问到,如果这段代码涉及到了共享资源(数据)就需要保证数据的正确性.也就是所谓的线程安全.之前 ...

  5. java中ReentrantReadWriteLock读写锁的使用

    Lock比传统线程模型中的synchronized方式更加面向对象,与生活中的锁类似,锁本身也应该是一个对象.两个线程执行的代码片段要实现同步互斥的效果,它们必须用同一个Lock对象. 读写锁:分为读 ...

  6. Java中的读写锁

    一.读写锁 1.初识读写锁 a)Java中的锁——Lock和synchronized中介绍的ReentrantLock和synchronized基本上都是排它锁,意味着这些锁在同一时刻只允许一个线程进 ...

  7. Java多线程,对锁机制的进一步分析

    1 可重入锁 可重入锁,也叫递归锁.它有两层含义,第一,当一个线程在外层函数得到可重入锁后,能直接递归地调用该函数,第二,同一线程在外层函数获得可重入锁后,内层函数可以直接获取该锁对应其它代码的控制权 ...

  8. JAVA线程锁-读写锁

    JAVA线程锁,除Lock的传统锁,又有两种特殊锁,叫读写锁ReadWriteLock 其中多个读锁不互斥,读锁和写锁互斥,写锁和写锁互斥 例子: /** * java线程锁分为读写锁 ReadWri ...

  9. java并发编程-读写锁

    最近项目中需要用到读写锁 读写锁适用于读操作多,写操作少的场景,假设你的程序中涉及到对一些共享资源的读和写操作,且写操作没有读操作那么频繁.在没有写操作的时候,两个线程同时读一个资源没有任何问题,所以 ...

随机推荐

  1. python 中的省略号

    在查看django源码时遇到下列内容:sweat: 这个省略号是什么意思? 来自为知笔记(Wiz)

  2. Linux命令查看各端口号占用情况

    yum -y install net-tools netstat -ntpl

  3. spring的返回json数据给前端注意事项

    1. 记得加入SpringJSON解析包,否则spring无法将返回的pojo类解析成json数据 maven则加上下面的依赖包 <!-- SpringJSON解析包--> <jac ...

  4. java 线程 总结

    1.前言 (1)线程的上一级是进程,进程是程序的一次执行过程,是系统运行程序的基本单位,因此进程是动态的. (2)线程与进程相似,但线程是一个比进程更小的执行单位,也被称为轻量级进程.一个进程在其执行 ...

  5. linux下玩转磁盘管理与挂载硬盘

    前言 本文将带来linux下的磁盘管理中的硬盘挂载,Linux操作系统挂载硬盘需要了解的一些知识.这可能是迄今为止介绍的最最最实用的linux硬盘挂载的文章了,比较详细.由于工作原因,平时使用的比较多 ...

  6. GLPK下载安装

    GLPK下载安装 下载 wget http://ftp.gnu.org/gnu/glpk/glpk-4.65.tar.gz tar -zxvf glpk-4.65.tar.gz 安装 如果你有管理员权 ...

  7. unity3d C# soket客户端接受失败

    using System.Collections; using System.Collections.Generic; using UnityEngine; using System; using S ...

  8. 如何在Xamarin中快速集成Android版认证服务-手机号码篇

    Xamarin作为微软提供的移动服务多系统开发平台,成为很多开发者首选的应用开发平台.AppGallery Connect(以下简称AGC)也在逐步的支持Xamarin的SDK.认证服务也是支持Xam ...

  9. LaTex用法笔记(一)——LaTex源文件的基本结构

    首先打开TeXstudio,创建一个新文件并保存 用\documentclass{article}引入一个文档类,也可以引用book/report/letter 然后用\begin{}和\end{}输 ...

  10. Tomcat-默认访问的工程和默认访问的资源

    Tomcat(默认访问的工程和默认访问的资源) ROOT的工程的访问,以及默认index.html页面的访问 当我们在浏览器地址栏中输入访问地址如下: http://ip:port/   ====== ...