服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - Java读写锁ReadWriteLock原理与应用场景详解

Java读写锁ReadWriteLock原理与应用场景详解

2023-02-26 15:08wdc Java教程

这篇文章主要介绍了Java读写锁ReadWriteLock原理与应用场景详解,读写状态的设计,写锁的获取与释放,锁降级需要的朋友可以参考下

Java并发编程提供了读写锁,主要用于读多写少的场景

 

什么是读写锁?

读写锁并不是JAVA所特有的读写锁(Readers-Writer Lock)顾名思义是一把锁分为两部分:读锁和写锁,其中读锁允许多个线程同时获得,因为读操作本身是线程安全的,而写锁则是互斥锁,不允许多个线程同时获得写锁,并且写操作和读操作也是互斥的。

所谓的读写锁(Readers-Writer Lock),顾名思义就是将一个锁拆分为读锁和写锁两个锁。

其中读锁允许多个线程同时获得,而写锁则是互斥锁,不允许多个线程同时获得写锁,并且写操作和读操作也是互斥的。

Java读写锁ReadWriteLock原理与应用场景详解

 

为什么需要读写锁?

Synchronized和 ReentrantLock 都是独占锁,即在同一时刻只有一个线程获取到锁。

然而在有些业务场景中,我们大多在读取数据,很少写入数据,这种情况下,如果仍使用独占锁,效率将及其低下。

针对这种情况,Java提供了读写锁——ReentrantReadWriteLock。

主要解决:对共享资源有读和写的操作,且写操作没有读操作那么频繁的场景。

 

读写锁的特点

  • 公平性:读写锁支持非公平和公平的锁获取方式,非公平锁的吞吐量优于公平锁的吞吐量,默认构造的是非公平锁
  • 可重入:在线程获取读锁之后能够再次获取读锁,但是不能获取写锁,而线程在获取写锁之后能够再次获取写锁,同时也能获取读锁
  • 锁降级:线程获取写锁之后获取读锁,再释放写锁,这样实现了写锁变为读锁,也叫锁降级

 

读写锁的使用场景

ReentrantReadWriteLock适合读多写少的场景:

读锁ReentrantReadWriteLock.ReadLock可以被多个线程同时持有, 所以并发能力很高。

写锁ReentrantReadWriteLock.WriteLock是独占锁, 在一个线程持有写锁时候, 其他线程都不能在抢占, 包含抢占读锁都会阻塞。

ReentrantReadWriteLock的使用场景总结:其实就是 读读并发、读写互斥、写写互斥而已,如果一个对象并发读的场景大于并发写的场景,那就可以使用 ReentrantReadWriteLock来达到保证线程安全的前提下提高并发效率。

 

读写锁的主要成员和结构图

1. ReentrantReadWriteLock的继承关系

Java读写锁ReadWriteLock原理与应用场景详解

public interface ReadWriteLock {
  /**
   * Returns the lock used for reading.
   *
   * @return the lock used for reading.
   */
  Lock readLock();
  /**
   * Returns the lock used for writing.
   *
   * @return the lock used for writing.
   */
  Lock writeLock();
}

读写锁 ReadWriteLock

读写锁维护了一对相关的锁,一个用于只读操作,一个用于写入操作。

只要没有写入,读取锁可以由多个读线程同时保持,写入锁是独占的。

2.ReentrantReadWriteLock的核心变量

Java读写锁ReadWriteLock原理与应用场景详解

ReentrantReadWriteLock类包含三个核心变量:

  • ReaderLock:读锁,实现了Lock接口
  • WriterLock:写锁,也实现了Lock接口
  • Sync:继承自AbstractQueuedSynchronize(AQS),可以为公平锁FairSync 或 非公平锁NonfairSync

3.ReentrantReadWriteLock的成员变量和构造函数

/** 内部提供的读锁 */

  private final ReentrantReadWriteLock.ReadLock readerLock;

  /** 内部提供的写锁 */
  private final ReentrantReadWriteLock.WriteLock writerLock;

  /** AQS来实现的同步器 */
  final Sync sync;

  /**
   * Creates a new {@code ReentrantReadWriteLock} with
   * 默认创建非公平的读写锁
   */
  public ReentrantReadWriteLock() {
      this(false);
  }

  /**
   * Creates a new {@code ReentrantReadWriteLock} with
   * the given fairness policy.
   *
   * @param fair {@code true} if this lock should use a fair ordering policy
   */
  public ReentrantReadWriteLock(boolean fair) {
      sync = fair ? new FairSync() : new NonfairSync();
      readerLock = new ReadLock(this);
      writerLock = new WriteLock(this);
  }

 

读写锁的实现原理

ReentrantReadWriteLock实现关键点,主要包括:

  • 读写状态的设计
  • 写锁的获取与释放
  • 读锁的获取与释放
  • 锁降级

1.读写状态的设计

之前谈ReentrantLock的时候,Sync类是继承于AQS,主要以int state为线程锁状态,0表示没有被线程占用,1表示已经有线程占用。

同样ReentrantReadWriteLock也是继承于AQS来实现同步,那int state怎样同时来区分读锁和写锁的?

如果在一个整型变量上维护多种状态,就一定需要“按位切割使用”这个变量,ReentrantReadWriteLock将int类型的state将变量切割成两部分:

  • 高16位记录读锁状态
  • 低16位记录写锁状态

Java读写锁ReadWriteLock原理与应用场景详解

abstract static class Sync extends AbstractQueuedSynchronizer {
  // 版本序列号
  private static final long serialVersionUID = 6317671515068378041L;        
  // 高16位为读锁,低16位为写锁
  static final int SHARED_SHIFT   = 16;
  // 读锁单位
  static final int SHARED_UNIT    = (1 << SHARED_SHIFT);
  // 读锁最大数量
  static final int MAX_COUNT      = (1 << SHARED_SHIFT) - 1;
  // 写锁最大数量
  static final int EXCLUSIVE_MASK = (1 << SHARED_SHIFT) - 1;
  // 本地线程计数器
  private transient ThreadLocalHoldCounter readHolds;
  // 缓存的计数器
  private transient HoldCounter cachedHoldCounter;
  // 第一个读线程
  private transient Thread firstReader = null;
  // 第一个读线程的计数
  private transient int firstReaderHoldCount;
}

2.写锁的获取与释放

protected final boolean tryAcquire(int acquires) {
          /*
           * Walkthrough:
           * 1. If read count nonzero or write count nonzero
           *    and owner is a different thread, fail.
           * 2. If count would saturate, fail. (This can only
           *    happen if count is already nonzero.)
           * 3. Otherwise, this thread is eligible for lock if
           *    it is either a reentrant acquire or
           *    queue policy allows it. If so, update state
           *    and set owner.
           */
          Thread current = Thread.currentThread();
          int c = getState();
          //获取独占锁(写锁)的被获取的数量
          int w = exclusiveCount(c);
          if (c != 0) {
              // (Note: if c != 0 and w == 0 then shared count != 0)
              //1.如果同步状态不为0,且写状态为0,则表示当前同步状态被读锁获取
              //2.或者当前拥有写锁的线程不是当前线程
              if (w == 0 || current != getExclusiveOwnerThread())
                  return false;
              if (w + exclusiveCount(acquires) > MAX_COUNT)
                  throw new Error("Maximum lock count exceeded");
              // Reentrant acquire
              setState(c + acquires);
              return true;
          }
          if (writerShouldBlock() ||
              !compareAndSetState(c, c + acquires))
              return false;
          setExclusiveOwnerThread(current);
          return true;
      }

1)c是获取当前锁状态,w是获取写锁的状态。

2)如果锁状态不为零,而写锁的状态为0,则表示读锁状态不为0,所以当前线程不能获取写锁。或者锁状态不为零,而写锁的状态也不为0,但是获取写锁的线程不是当前线程,则当前线程不能获取写锁。

3)写锁是一个可重入的排它锁,在获取同步状态时,增加了一个读锁是否存在的判断。

写锁的释放与ReentrantLock的释放过程类似,每次释放将写状态减1,直到写状态为0时,才表示该写锁被释放了。

3.读锁的获取与释放

protected final int tryAcquireShared(int unused) {
  for(;;) {
      int c = getState();
      int nextc = c + (1<<16);
      if(nextc < c) {
         throw new Error("Maxumum lock count exceeded");
      }
      if(exclusiveCount(c)!=0 && owner != Thread.currentThread())
         return -1;
      if(compareAndSetState(c,nextc))
         return 1;
  }
}

1)读锁是一个支持重进入的共享锁,可以被多个线程同时获取。

2)在没有写状态为0时,读锁总会被成功获取,而所做的也只是增加读状态(线程安全)

3)读状态是所有线程获取读锁次数的总和,而每个线程各自获取读锁的次数只能选择保存在ThreadLocal中,由线程自身维护。

读锁的每次释放均减小状态(线程安全的,可能有多个读线程同时释放锁),减小的值是1<<16。

4.锁降级

降级是指当前把持住写锁,再获取到读锁,随后释放(先前拥有的)写锁的过程。

锁降级过程中的读锁的获取是否有必要,答案是必要的。主要是为了保证数据的可见性,如果当前线程不获取读锁而直接释放写锁,假设此刻另一个线程获取的写锁,并修改了数据,那么当前线程就步伐感知到线程T的数据更新,如果当前线程遵循锁降级的步骤,那么线程T将会被阻塞,直到当前线程使数据并释放读锁之后,线程T才能获取写锁进行数据更新。

5.读锁与写锁的整体流程

Java读写锁ReadWriteLock原理与应用场景详解

 

读写锁总结

本篇详细介绍了ReentrantReadWriteLock的特征、实现、锁的获取过程,通过4个关键点的核心设计:

  • 读写状态的设计
  • 写锁的获取与释放
  • 读锁的获取与释放
  • 锁降级

从而才能实现:共享资源有读和写的操作,且写操作没有读操作那么频繁的应用场景。

以上就是Java读写锁ReadWriteLock原理与应用场景详解的详细内容,更多关于Java读写锁ReadWriteLock原理与应用场景的资料请关注服务器之家其它相关文章!

原文链接:https://mikechen.cc/17389.html

延伸 · 阅读

精彩推荐
  • Java教程Java实现简单的贪吃蛇小游戏

    Java实现简单的贪吃蛇小游戏

    这篇文章主要为大家详细介绍了Java实现简单的贪吃蛇小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    just sh!10782021-09-08
  • Java教程Java你不了解的大数型BigInteger与BigDecimal类

    Java你不了解的大数型BigInteger与BigDecimal类

    这篇文章主要介绍了Java 处理超大数类型之BigInteger与BigDecimal案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋...

    星辰与晨曦5932022-12-14
  • Java教程Java简易计算器程序设计

    Java简易计算器程序设计

    这篇文章主要介绍了Java简易计算器程序设计的相关参考资料,需要的朋友可以参考下 ...

    IT_xiao小巫2132020-01-09
  • Java教程java对象与json对象间的相互转换的方法

    java对象与json对象间的相互转换的方法

    本篇文章主要介绍了java对象与json对象间的相互转换的方法,详细介绍了json字符串和java对象相互转换,有兴趣的可以了解一下...

    gchb952718382020-07-23
  • Java教程Java中的对象流总结(必看篇)

    Java中的对象流总结(必看篇)

    下面小编就为大家带来一篇Java中的对象流总结(必看篇)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    Java之家4322020-09-22
  • Java教程Java设计模式之外观模式示例详解

    Java设计模式之外观模式示例详解

    外观模式为多个复杂的子系统,提供了一个一致的界面,使得调用端只和这个接口发生调用,而无须关系这个子系统内部的细节。本文将通过示例详细为大...

    温故知新之java5312022-08-16
  • Java教程归并排序的原理及java代码实现

    归并排序的原理及java代码实现

    归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的。然后再把有序子序...

    hebedich5002020-03-26
  • Java教程java反射机制最详解

    java反射机制最详解

    这篇文章主要介绍了Java 反射机制原理与用法,结合实例形式详细分析了Java反射机制的相关概念、原理、基本使用方法及操作注意事项,需要的朋友可以参考...

    勇敢牛牛不怕困难@帅6512021-12-03