读写锁

读写锁在同一时刻可以允许多个读线程访问,但是在写线程访问时,所有的读线程和其他写线程均被阻塞。读写锁维护了一对锁,一个读锁和一个写锁,通过分离读锁和写锁,使得并发性相比一般的排他锁有了很大提升。

读操作可以多个线程,写操作只能一个线程

Java并发包提供读写锁的实现是 ReentrantReadWriteLock

特性:

  1. 支持公平性和非公平的锁获取方式
  2. 支持重进入:以读写线程为例,当读线程获取读锁以后,还能再次获取读锁,而写线程在获取写锁时还未完全释放的时候还能再获取写锁以及也能获取读锁。
  3. 锁降级。写锁可以降级为读锁,但是读锁不能升级为写锁

锁降级的定义:

锁降级指的是写锁降级成为读锁。如果当前线程拥有写锁,然后将其释放,最后再获取读锁,这种分段完成的过程不能称之为锁降级。锁降级是指把持住(当前拥有的)写锁,再获取到读锁,随后释放(先前拥有的)写锁的过程。

写锁可以降级为读锁顺序:获取写锁----获取读锁------释放写锁------释放读锁。

其缺点:会造成锁饥饿问题 一直读,没有写操作。

资源与锁的三个状态:

  1. 无锁,多线程抢夺资源 乱
  2. 添加锁(Synchronized和ReentrantLock) 都是独占,读读、读写、写写都是独占,每次只能一个操作
  3. 读写锁,读读可以共享,提升性能,同时可以多人进行读操作

ReentrantReadWriteLock 目的就是:提高读操作的吞吐量 (可用于读多写少的情况下)

读写锁可重入的理解:

读锁的重入是允许多个申请读操作的线程,而写锁同时只能允许单个线程占有,该线程的写操作可以重入。

如果一个线程占有了写锁,在不释放写锁的情况下,它还能占有读锁,也就是锁的降级。

如果一个线程同时占有了读锁和写锁,在完全释放了写锁,那么就转换为了读锁,以后写操作无法重入,如果写锁未完全释放时,写操作时可以重入的。

失败例子:

package com.RWLock;

import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.TimeUnit; class MyCache{
private volatile Map<String,Object> map = new HashMap<String,Object>(); //写操作
public void put(String key,Object value) throws InterruptedException {
System.out.println(Thread.currentThread().getName()+"\t------写入数据"+key);
TimeUnit.SECONDS.sleep(1);
map.put(key,value);
System.out.println(Thread.currentThread().getName()+"\t------写入完成"+key);
}
//读操作
public void get(String key) throws InterruptedException {
System.out.println(Thread.currentThread().getName()+"\t------读数据"+key);
TimeUnit.SECONDS.sleep(1);
map.get(key);
System.out.println(Thread.currentThread().getName()+"\t------读取完成"+key);
}
} public class readWriteLockDemo {
public static void main(String[] args) {
MyCache myCache = new MyCache();
//多个线程进行写操作
for (int i = 1; i <= 5; i++) {
int finalI = i;
new Thread(()->{
try {
myCache.put(finalI +"", finalI +"");
} catch (InterruptedException e) {
e.printStackTrace();
}
},String.valueOf(i)).start();
}
for (int i = 1; i <= 5; i++) {
int finalI = i;
new Thread(()->{
try {
myCache.get(finalI+"");
} catch (InterruptedException e) {
e.printStackTrace();
}
},String.valueOf(i)).start();
}
}
}

使用读写锁以后:

Cache组合一个非线程安全的HashMap作为缓存的实现,同时使用读写锁的读锁和写锁来保证Cache是线程安全的。在读操作get(String key)方法中,需要获取读锁,这使得并发访问该方法时不会被阻塞。写操作put(String key,Object value)方法和clear()方法,在更新HashMap时必须提前获取写锁,当获取写锁后,其他线程对于读锁和写锁的获取均被阻塞,而只有写锁被释放之后,其他读写操作才能继续。

package com.RWLock;

import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock; class MyCache{
private volatile Map<String,Object> map = new HashMap<String,Object>();
//可重入的读写锁
private ReadWriteLock readWriteLock = new ReentrantReadWriteLock(); //写操作
public void put(String key,Object value) throws InterruptedException {
try{
readWriteLock.writeLock().lock(); //写锁
System.out.println(Thread.currentThread().getName()+"\t------写入数据"+key);
TimeUnit.SECONDS.sleep(1);
map.put(key,value);
System.out.println(Thread.currentThread().getName()+"\t------写入完成"+key);
}catch (InterruptedException e){
e.printStackTrace();
}finally{
readWriteLock.writeLock().unlock();
} }
//读操作
public void get(String key) {
try {
readWriteLock.readLock().lock(); //读锁
System.out.println(Thread.currentThread().getName()+"\t------读数据"+key);
TimeUnit.SECONDS.sleep(1);
map.get(key);
System.out.println(Thread.currentThread().getName()+"\t------读取完成"+key);
}catch (InterruptedException e){
e.printStackTrace();
}finally{
readWriteLock.readLock().unlock();
} }
} public class readWriteLockDemo {
public static void main(String[] args) {
MyCache myCache = new MyCache();
//多个线程进行写操作
for (int i = 1; i <= 5; i++) {
int finalI = i;
new Thread(()->{
try {
myCache.put(finalI +"", finalI +"");
} catch (Exception e) {
e.printStackTrace();
}
},String.valueOf(i)).start();
}
for (int i = 1; i <= 5; i++) {
int finalI = i;
new Thread(()->{
try {
myCache.get(finalI+"");
} catch (Exception e) {
e.printStackTrace();
}
},String.valueOf(i)).start();
}
}
}

读写锁的设计:依赖于同步器的同步状态实现的。

同步状态表示锁被一个线程重复获取的次数,而读写锁的自定义同步器需要在同步状态(一个整型变量)上维护多个读线程和一个写线程的状态,使得该状态的设计成为读写锁实现的关键

如果一个整型变量维护,按位切割,高16位为读状态,低16位为写状态。

读写锁是如何迅速确定读和写各自的状态呢?答案是通过位运算

写锁的获取和释放:

写锁是一个支持重进入的排他锁;

  1. 如果当前线程获取了写锁,则增加写状态,独占
  2. 如果当前线程(A)再获取锁时,读锁已经被获取或者该线程不是已经获取写锁的线程(个人理解:如果有线程获取了写锁,则其他读写线程的后续访问均被阻塞),则当前线程(A)进入等待状态。

获取读锁后不能获取写锁,但是获取写锁后可以获取读锁

读锁的获取和释放

读锁是一个支持重进入的共享锁,它能够被多个线程同时获取,在没有其他写线程访问(或者写状态为0)时,读锁总会被成功地获取,而所做的也只是(线程安全的)增加读状态。

如果其他线程已经获取了写锁,则当前线程获取读锁失败,进入等待状态。如果当前线程获取了写锁或者写锁未被获取,则当前线程(线程安全,依靠CAS保证)增加读状态,成功获取读锁

JUC之读写锁问题的更多相关文章

  1. JUC——线程同步锁(ReentrantReadWriteLock读写锁)

    读写锁简介 所谓的读写锁值得是两把锁,在进行数据写入的时候有一个把“写锁”,而在进行数据读取的时候有一把“读锁”. 写锁会实现线程安全同步处理操作,而读锁可以被多个对象读取获取. 读写锁:ReadWr ...

  2. JUC 并发编程--04 常用的辅助类CountDownLatch , CyclicBarrier , Semaphore , 读写锁 , 阻塞队列,CompletableFuture(异步回调)

    CountDownLatch 相当于一个减法计数器, 构造方法指定一个数字,比如6, 一个线程执行一次,这个数字减1, 当变为0 的时候, await()方法,才开始往下执行,, 看这个例子 Cycl ...

  3. ReadWriteLock读写锁(八)

    前言:在JUC ReentrantReadWriteLock是基于AQS实现的读写锁实现. ReadWriteLock中定义了读写锁需要实现的接口,具体定义如下: public interface R ...

  4. AQS系列(四)- ReentrantReadWriteLock读写锁的释放锁

    前言 继续JUC包中ReentrantReadWriteLock的学习,今天学习释放锁. 一.写锁释放锁 入口方法 public void unlock() { sync.release(1); } ...

  5. JUC-10-ReadWriteLock读写锁

    ReadWriteLock同Lock一样也是一个接口,提供了readLock和writeLock两种锁的操作机制,一个是只读的锁,一个是写锁  

  6. 读写锁ReentrantReadWriteLock源代码浅析

    1.简介 并发中常用的ReentrantLock,是一种典型的排他锁,这类锁在同一时刻只允许一个线程进行访问,实际上将并行操作变成了串行操作.在并发量大的业务中,其整体效率.吞吐量不能满足实现的需要. ...

  7. 读-写锁 ReadWriteLock & 线程八锁

    读-写锁 ReadWriteLock: ①ReadWriteLock 维护了一对相关的锁,一个用于只读操作, 另一个用于写入操作. 只要没有 writer,读取锁可以由 多个 reader 线程同时保 ...

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

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

  9. java多线程-读写锁

    Java5 在 java.util.concurrent 包中已经包含了读写锁.尽管如此,我们还是应该了解其实现背后的原理. 读/写锁的 Java 实现(Read / Write Lock Java ...

随机推荐

  1. 使用.NET 6开发TodoList应用(1)——系列背景

    前言 想到要写这样一个系列博客,初衷有两个:一是希望通过一个实践项目,将.NET 6 WebAPI开发的基础知识串联起来,帮助那些想要入门.NET 6服务端开发的朋友们快速上手,对使用.NET 6开发 ...

  2. C51单片机0~60计数器

    源码 #include<reg51.h> unsigned char code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f, ...

  3. 使用react搭建组件库:react+typescript+storybook

    前期准备 1. 初始化项目 npx create-react-app react-components --template typescript 2. 安装依赖 使用哪种打包方案:webpack/r ...

  4. JAVA从URL参数链接中获取指定参数的值

    import java.util.HashMap; import java.util.Map; /** * @author yvioo */ public class UrlUtils { /** * ...

  5. 【LeetCode】257. Binary Tree Paths 解题报告(java & python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 递归 迭代 日期 题目地址:https://leet ...

  6. codeforces 624C Graph and String

    C. Graph and String time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

  7. 一个网关服务性能问题的Dump分析

    本篇文章分为三部分,首先简单介绍一下分析的工具Windbg,其次针对一个网关服务性能问题进行逐步刨析,最后针对性能问题的分析总结. 一 Windbg介绍 1.Windbg是个非常强大的调试器,它设计了 ...

  8. Java 泛型通配符 T,E,K,V,?

    Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许开发者在编译时检测到非法的类型. 泛型的本质是参数化类型,也就是说所操作的数据类型被 ...

  9. 遍历 HashMap 的 5 种最佳方式

    使用 Iterator 遍历 HashMap EntrySet 使用 Iterator 遍历 HashMap KeySet 使用 For-each 循环迭代 HashMap 使用 Lambda 表达式 ...

  10. Regularizing Deep Networks with Semantic Data Augmentation

    目录 概 主要内容 代码 Wang Y., Huang G., Song S., Pan X., Xia Y. and Wu C. Regularizing Deep Networks with Se ...