突击并发编程JUC系列演示代码地址:
https://github.com/mtcarpenter/JavaTutorial

本章节将学习 ReentrantReadWriteLock(读写锁),ReadWriteLock 也是 java 5之后引入的,之前提到锁(如MutexReentrantLock)基本都是排他锁,这些锁在同一时刻只允许一个线程进行访问,而读写锁在同一时刻可以允许多个读线程访问,但是在写线程访问时,所有的读线程和其他写线程均被阻塞。读写锁维护了一对锁,一个读锁和一个写锁,通过分离读锁和写锁,使得并发性相比一般的排他锁有了很大提升。
读写锁的访问约束:

  • 读-读不互斥:读读之间不阻塞
  • 读-写互斥:读堵塞写,写也阻塞读
  • 写-写互斥:写写阻塞

ReadWriteLock

ReentrantReadWriteLockReadWriteLock接口的实现,ReadWriteLock仅定义了获取读锁和写锁的两个方法,即readLock()方法和writeLock()方法。

public interface ReadWriteLock {Lock readLock();Lock writeLock();
}

除了接口方法之外,ReentrantReadWriteLock 还提供了一些便于外界监控其内部工作状态的方法。

  • int getReadLockCount(): 返回当前读锁被获取的次数。该次数不等于获取读锁的线程数,例如,仅一个线程,他连续获取(重进入)了 n 次读锁,那么占据读锁的线程数是 1 ,但该方法返回 n。
  • int getReadHoldCount():返回当前线程获取读锁的次数。该方法在 Java 6 中加入到 ReentrantReadWriteLock 中,使用 ThreadLocal 保存当前线程获取次数,这也使得 Java 6 的实现变得更加复杂
  • boolean isWriteLocked() : 判断写锁是否被获取
  • int getWriteHoldCount(): 返回当前写锁被获取的次数

ReentrantReadWriteLock

通过 ReentrantReadWriteLock 实现一个简单的缓存,代码示例如下:

public class LockExample3 {private static final Map<String, Object> map = new HashMap<>();private static final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();private static final Lock readLock = lock.readLock();private static final Lock writeLock = lock.writeLock();/*** 向 map 存入数据** @param key* @param value* @return*/public static Object put(String key, Object value) {writeLock.lock();try {return map.put(key, value);} finally {writeLock.unlock();}}/*** 获取单个键值的值** @param key* @return*/public static Object get(String key) {readLock.lock();try {return map.get(key);} finally {readLock.unlock();}}/*** 获取 map 的键值** @return*/public static Set<String> getAllKeys() {readLock.lock();try {return map.keySet();} finally {readLock.unlock();}}/*** 清除 map 所有的数据*/public static void clear() {writeLock.lock();try {map.clear();} finally {writeLock.unlock();}}}

测试代码

public class LockExample3Test {// 请求总数public static int requestTotal = 10;public static void main(String[] args) throws Exception {ExecutorService executorService = Executors.newCachedThreadPool();final CountDownLatch countDownLatch = new CountDownLatch(requestTotal);for (int i = 0; i < requestTotal; i++) {final String temp = String.valueOf(i);executorService.execute(() -> {try {add(temp);} catch (Exception e) {}countDownLatch.countDown();});}// 等待所有的线程运行完成countDownLatch.await();// 多线程获取 keyfor (int i = 0; i < requestTotal; i++) {final String temp = String.valueOf(i);executorService.execute(() -> {try {get(temp);} catch (Exception e) {}});}executorService.shutdown();TimeUnit.SECONDS.sleep(1);// 获取所有的keysSystem.out.println("获取所有的键值\t" + LockExample3.getAllKeys());//  清除所有的 keysLockExample3.clear();// 再次获取所有的keys 发现已被清空System.out.println("获取所有的键值\t" + LockExample3.getAllKeys());}private static void add(String i) {LockExample3.put(i, Thread.currentThread().getName());}private static void get(String i) {System.out.println(i + "\t" + LockExample3.get(i));}
}

运行结果如下:

0    pool-1-thread-1
1   pool-1-thread-2
2   pool-1-thread-3
4   pool-1-thread-5
3   pool-1-thread-4
8   pool-1-thread-9
7   pool-1-thread-8
6   pool-1-thread-7
5   pool-1-thread-6
9   pool-1-thread-10
获取所有的键值 [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
获取所有的键值 []

欢迎关注公众号 山间木匠 , 我是小春哥,从事 Java 后端开发,会一点前端、通过持续输出系列技术文章以文会友,如果本文能为您提供帮助,欢迎大家关注、 点赞、分享支持,我们下期再见!

突击并发编程JUC系列-ReentrantReadWriteLock相关推荐

  1. atomiclong 初始化_突击并发编程JUC系列-原子更新AtomicLong

    Java 从JDK 1.5开始提供了java.util.concurrent.atomic包(以下简称Atomic包),这个包中的原子操作类提供了一种用法简单.性能高效.线程安全地更新一个变量的方式. ...

  2. 并发编程JUC系列及部分问题

    什么是 CAS 吗? CAS(Compare And Swap)指比较并交换.CAS算法CAS(V, E, N)包含 3 个参数,V 表示要更新的变量,E 表示预期的值,N 表示新值.在且仅在 V 值 ...

  3. 多线程进阶(并发编程JUC)

    多线程进阶(并发编程JUC) 提示: 本材料只做个人学习参考,不作为系统的学习流程,请注意识别!!! 并发编程JUC 1. 基础知识 什么是JUC(Java并发包)?并发编程的本质(充分利用CPU的资 ...

  4. java lock 对象_Java并发编程锁系列之ReentrantLock对象总结

    Java并发编程锁系列之ReentrantLock对象总结 在Java并发编程中,根据不同维度来区分锁的话,锁可以分为十五种.ReentranckLock就是其中的多个分类. 本文主要内容:重入锁理解 ...

  5. 并发编程——JUC并发编程知识脑图

    摘要 并发编程在软件编程中尤为突出和重要,在当今面试或工作中也是不可缺少的.作为一名高级java开发工程师,并发编程的技能已经成为了重要的一项.本博文将详细介绍并发编程中的知识点和知识脑图,帮助大家更 ...

  6. 多线程 可参考 博主【 lx青萍之末】 的 【C++并发编程 】系列博客

    关于多线程 相关知识 可参考博主[ lx青萍之末] 的 [C++并发编程 ]系列博客 https://blog.csdn.net/daaikuaichuan/category_6887432.html

  7. Java并发编程—JUC的Lock锁

    一.Lock (JUC锁) JUC 锁位于java.util.concurrent.locks包下,为锁和等待条件提供一个框架,它不同于内置同步和监视器. CountDownLatch,CyclicB ...

  8. 并发编程——JUC并发大厂面试问题

    摘要 现如今,不管是应届毕业生还是工作了三五年之内的工程师,在面试招聘的时候JUC并发编程的是必须掌握的一个技能,否者你将会被面试官玩弄.本博文将整理有关于的JUC的大厂面试问题和答案.帮助大家在面试 ...

  9. JAVA并发编程JUC基础学习(简介)

    2019独角兽企业重金招聘Python工程师标准>>> 之前写过一篇并发编程的简单实例应用,Future快速实现并发编程,可以很快的在自己的项目中应用,但并不系统,之前说过总结一篇( ...

最新文章

  1. django1.4日志模块配置及使用
  2. zabbix web前端取值同后端取值不一致
  3. 《深入理解计算机系统》读书笔记七:浮点数表示
  4. BigData之Hadoop:Hadoop的简介、深入理解、下载、案例应用之详细攻略
  5. 抗衰老,吃这些食物越来越年轻
  6. 大型综合体弱电智能化解决方案标书
  7. 几种实用的pythonic语法
  8. 链路追踪Skywalking保姆级安装教程
  9. 基于Caffe ResNet-50网络实现图片分类(仅推理)的实验复现
  10. 二维邮局选址问题-带权中位数
  11. 产品标题什么时候进行优化,提高权重,标题优化的技巧方法
  12. “空天地海”一体化的海上应急通信网络技术综述
  13. 从零开始学习大数据系列之Linux-02Vim与Shell script
  14. 【源译】Optimizeit Profiler概览
  15. 利用Python进行数据分析第二版复现(五)
  16. 简述计算机系统集成的特点,谈计算机系统集成的特点与发展
  17. 视角设置(第一人称、第三人称)
  18. cesium类文档查找
  19. 电子化工作方式:方便保存,提高效率
  20. 全球知名图像研究机构网站

热门文章

  1. Chrome 截图页面全图
  2. 把握问题的关键(转自知乎)
  3. Java基础加强重温_05:Iterator迭代器、增强for循环、集合综合案例-斗地主、数据结构(栈、队列、数组、链表、红黑树)、List接口、Set接口
  4. idea把mybatis的sql黄色背景去掉
  5. 不修复软件缺陷的原因
  6. 动态语言、静态语言、脚本语言、解释型语言、编译型语言
  7. 安卓手机上玩口袋妖怪gba全集的方法
  8. 微信小程序开发之——import、require和include
  9. Linux登录oracle
  10. 修改IDEA的堆内存