读写锁

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

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

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. not_the_same_3dsctf_2016

    老样子查看程序开启的保护 可以看到程序是32位的程序开启了nx保护,把程序放入ida编译一下 shift+f12可以看到flag.txt,我们用ctrl+x跟随一下 看到程序,直接想到的就是通过溢出获 ...

  2. C# VS 调试报错:未能加载文件或程序集“Newtonsoft.Json, Version=4.5.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed”或它的某一个依赖项

    今天在使用 VS(VisualStudio) 调试一个复杂的 WinForm 程序,总是提示错误: 未能加载文件或程序集"Newtonsoft.Json, Version=4.5.0.0, ...

  3. 小迪安全 Web安全 基础入门 - 第二天 - Web应用&架构搭建&漏洞&HTTP数据包&代理服务器

    一.网站搭建 1.域名.是由一串用点分隔的字符组成的互联网上某一台计算机或计算机组的名称,用于在数据传输时标识计算机的电子方位.域名可以说是一个IP地址的代称,目的是为了便于记忆后者. 2.子域名.在 ...

  4. 【蓝桥杯】非VIP基础题型训练17题 (Python 题解)

    NO.I 基础题型 基础练习汇总 时间 题目 解析 21.12.24 早上 1. A+B问题 练习系统的适应 21.12.24 早上 2. 数组排序 输入输出排序 21.12.24 早上 3. 十六进 ...

  5. 【LeetCode】999. Available Captures for Rook 解题报告(C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 暴力遍历 日期 题目地址:https://leetc ...

  6. 【LeetCode】500. Keyboard Row 解题报告(Java & Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 暴力解 字典 + set 日期 题目地址:https ...

  7. 晴天小猪历险记之Hill(Dijkstra优先队列优化)

    描述 这一天,他来到了一座深山的山脚下,因为只有这座深山中的一位隐者才知道这种药草的所在.但是上山的路错综复杂,由于小小猪的病情,晴天小猪想找一条需时最少的路到达山顶,但现在它一头雾水,所以向你求助. ...

  8. struts2升级至2.3.24方法

    1.替换如下jar包 2.修改web.xml中的struts过滤器配置 将原来的过滤配置注释掉 替换为: 3.struts.xml配置 4. 发现程序中有类报错:缺少  import org.apac ...

  9. LeetCode—剑指 Offer学习计划

    第 1 天 栈与队列(简单) 剑指 Offer 09. 用两个栈实现队列 class CQueue { public: CQueue() { } stack<int>s1,s2; void ...

  10. leetcode日记本

    写在前面: 2019.6开始经过一年的学习,我依然没有学会算法,依然停留在最基本的阶段,面对题目依然一头雾水 但是难不是放弃的理由,根据毛主席的论持久战原理,我决定一天看一点循序渐进,相信总有一天可以 ...