首页 > 其他分享 >面试官:说说读写锁的实现原理?

面试官:说说读写锁的实现原理?

时间:2024-08-12 17:30:08浏览次数:13  
标签:面试官 Thread ReentrantReadWriteLock 写锁 读写 读锁 线程 原理

在实际项目开发中,并发编程一定会用(提升程序的执行效率),而用到并发编程那么锁机制就一定会用,因为锁是保证并发编程的主要手段。

在 Java 中常用的锁有以下几个:

  1. synchronized(内置锁):Java 语言内置的关键字,JVM 层级锁实现,使用起来较为简单直观。
  2. ReentrantLock(可重入锁):需要显式地获取和释放锁,提供了更灵活的锁操作方式。
  3. ReentrantReadWriteLock(读写锁):性能较好,分为读锁和写锁,允许多个读线程同时获取读锁,而写锁具有排他性。
  4. StampedLock(邮戳锁):JDK 8 提供的锁,提供了一种乐观读的方式,先尝试读取,如果在读取过程中没有发生写操作,则可以直接完成读取,避免了获取读锁的开销。

而我们今天重点要讨论的是读写锁 ReentrantReadWriteLock 和它的实现原理。

1.读写锁介绍

ReentrantReadWriteLock(读写锁)是 Java 并发包(java.util.concurrent.locks)中的一个类,它实现了一个可重入的读写锁。读写锁允许多个线程同时读取共享资源,但在写入共享资源时只允许一个线程进行

它把锁分为两部分:读锁和写锁,其中读锁允许多个线程同时获得,因为读操作本身是线程安全的,而写锁则是互斥锁,不允许多个线程同时获得写锁,并且写操作和读操作也是互斥的。

也就是说读写锁的特征是:

  1. 读-读操作不加锁。
  2. 读-写操作加锁。
  3. 写-写操作加锁。

2.基本使用

ReentrantReadWriteLock 锁分为以下两种:

  1. ReentrantReadWriteLock.ReadLock 表示读锁:它提供了 lock 方法进行加锁、unlock 方法进行解锁。
  2. ReentrantReadWriteLock.WriteLock 表示写锁:它提供了 lock 方法进行加锁、unlock 方法进行解锁。

它的基础使用如下代码所示:

// 创建读写锁
final ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();
// 获得读锁
final ReentrantReadWriteLock.ReadLock readLock = readWriteLock.readLock();
// 获得写锁
final ReentrantReadWriteLock.WriteLock writeLock = readWriteLock.writeLock();
// 读锁使用
readLock.lock();
try {
    // 业务代码...
} finally {
    readLock.unlock();
}
// 写锁使用
writeLock.lock();
try {
    // 业务代码...
} finally {
    writeLock.unlock();
}

2.1 读读不互斥

多个线程可以同时获取到读锁,称之为读读不互斥,如下代码所示:

// 创建读写锁
final ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();
// 创建读锁
final ReentrantReadWriteLock.ReadLock readLock = readWriteLock.readLock();
Thread t1 = new Thread(() -> {
    readLock.lock();
    try {
        System.out.println("[t1]得到读锁.");
        Thread.sleep(3000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    } finally {
        System.out.println("[t1]释放读锁.");
        readLock.unlock();
    }
});
t1.start();
Thread t2 = new Thread(() -> {
    readLock.lock();
    try {
        System.out.println("[t2]得到读锁.");
        Thread.sleep(3000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    } finally {
        System.out.println("[t2]释放读锁.");
        readLock.unlock();
    }
});
t2.start();

以上程序执行结果如下:
image.png

2.2 读写互斥

读锁和写锁同时使用是互斥的(也就是不能同时获得),这称之为读写互斥,如下代码所示:

// 创建读写锁
final ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();
// 创建读锁
final ReentrantReadWriteLock.ReadLock readLock = readWriteLock.readLock();
// 创建写锁
final ReentrantReadWriteLock.WriteLock writeLock = readWriteLock.writeLock();
// 使用读锁
Thread t1 = new Thread(() -> {
    readLock.lock();
    try {
        System.out.println("[t1]得到读锁.");
        Thread.sleep(3000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    } finally {
        System.out.println("[t1]释放读锁.");
        readLock.unlock();
    }
});
t1.start();
// 使用写锁
Thread t2 = new Thread(() -> {
    writeLock.lock();
    try {
        System.out.println("[t2]得到写锁.");
        Thread.sleep(3000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    } finally {
        System.out.println("[t2]释放写锁.");
        writeLock.unlock();
    }
});
t2.start();

以上程序执行结果如下:
image.png

2.3 写写互斥

多个线程同时使用写锁也是互斥的,这称之为写写互斥,如下代码所示:

// 创建读写锁
final ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();
// 创建写锁
final ReentrantReadWriteLock.WriteLock writeLock = readWriteLock.writeLock();
Thread t1 = new Thread(() -> {
    writeLock.lock();
    try {
        System.out.println("[t1]得到写锁.");
        Thread.sleep(3000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    } finally {
        System.out.println("[t1]释放写锁.");
        writeLock.unlock();
    }
});
t1.start();

Thread t2 = new Thread(() -> {
    writeLock.lock();
    try {
        System.out.println("[t2]得到写锁.");
        Thread.sleep(3000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    } finally {
        System.out.println("[t2]释放写锁.");
        writeLock.unlock();
    }
});
t2.start();

以上程序执行结果如下:
image.png

2.4 优点分析

  1. 提高了程序执行性能:多个读锁可以同时执行,相比于普通锁在任何情况下都要排队执行来说,读写锁提高了程序的执行性能。
  2. 避免读到临时数据:读锁和写锁是互斥排队执行的,这样可以保证了读取操作不会读到写了一半的临时数据。

2.5 适用场景

读写锁适合多读少写的业务场景,此时读写锁的优势最大。

3.底层实现

ReentrantReadWriteLock 是基于 AbstractQueuedSynchronizer(AQS)实现的,AQS 以单个 int 类型的原子变量来表示其状态,并通过 CAS 操作来保证线程安全。

这点也通过 ReentrantReadWriteLock 源码发现,ReentrantReadWriteLock 中的公平锁继承了 AbstractQueuedSynchronizer(AQS):

而 ReentrantReadWriteLock 中的非公平锁继承了公平锁(公平锁继承了 AbstractQueuedSynchronizer):

所以可以看出 ReentrantReadWriteLock 其底层主要是通过 AQS 实现的。

4.AQS

AbstractQueuedSynchronizer(AQS)是 Java 并发包中的一个抽象类,位于 java.util.concurrent.locks 包中。它为实现依赖于“独占”和“共享”模式的阻塞锁和相关同步器提供了一个框架。

AQS 是许多高级同步工具的基础,例如 ReentrantLock、ReentrantReadWriteLock、CountDownLatch 和 Semaphore。

4.1 AQS 核心概念

AQS 中有两个最主要的内容:

  1. 同步状态(State):用于表示同步器的状态,例如锁的持有数量、资源的可用数量等。可以通过 getState()、setState() 和 compareAndSetState() 方法来操作。
  2. 等待队列(CLH 队列):由双向链表实现的等待线程队列。当线程获取同步状态失败时,会被封装成节点加入到等待队列中。

4.2 AQS 工作流程

AQS 工作流程主要分为以下两部分。

  1. 加锁与释放锁
    • 线程尝试获取同步状态,如果获取成功,则直接执行后续操作。
    • 如果获取失败,则将当前线程封装成节点加入等待队列,并阻塞当前线程。
    • 当持有锁的线程释放锁时,会唤醒等待队列中的后继节点线程,使其重新尝试获取锁。
  2. 等待与唤醒
    • 等待队列中的节点通过自旋和阻塞来等待被唤醒。
    • 唤醒操作会按照一定的规则选择等待队列中的节点进行唤醒。

课后思考

AQS 是如何实现独占锁和共享锁的?AQS 使用了什么设计模式?

本文已收录到我的面试小站 www.javacn.site,其中包含的内容有:Redis、JVM、并发、并发、MySQL、Spring、Spring MVC、Spring Boot、Spring Cloud、MyBatis、设计模式、消息队列等模块。

标签:面试官,Thread,ReentrantReadWriteLock,写锁,读写,读锁,线程,原理
From: https://www.cnblogs.com/vipstone/p/18355403

相关文章

  • BOOST c++库学习 之 boost.thread入门实战指南 使用boost.thread库以及读写锁mutex的
    Boost.Thread库简介1.概述Boost.Thread库是Boost库中专门用于处理多线程编程的模块。它提供了一组跨平台的线程管理和同步工具,帮助开发者在C++中更轻松地编写多线程程序。Boost.Thread的设计目标是使多线程编程更加简单、可靠,同时保持高效和可移植性。2.Boost.Thread......
  • FIR滤波器的原理
    信号通过一个FIR滤波器其实就是信号与FIR滤波器的系数进行卷积(即乘累加)的过程。我们以一个简单信号模型为例,了解一下FIR波形器的原理。1.给定三组信号现在有三组信号,分别是:信号1:低频信号,即在时域上变化慢的信号,其输入先后为11112222。信号2:直流信号,其输入先后为1111......
  • Java Reentrantlock可重入锁原理 | 源码探究
    一、基本概念ReentrantLock是Java中提供的一个可重入互斥锁,它是java.util.concurrent.locks包中的一个接口Lock的实现类。ReentrantLock提供了比使用synchronized关键字更强大的锁定机制,例如 公平锁 和 非公平锁 选择、尝试锁定、可中断锁定等。ReentrantLock......
  • linux反向代理原理:帮助用户更好地优化网络架构
    Linux反向代理原理详解反向代理是一种在网络架构中常用的技术,尤其在Linux环境下被广泛应用。它可以帮助实现负载均衡、安全防护和请求缓存等功能。本文将深入探讨Linux反向代理的原理、工作机制以及其应用场景。1.什么是反向代理反向代理是指代理服务器接收客户端的请求,......
  • api代理爬虫:了解其基本原理和使用方法
    ​API代理爬虫的使用指南在数据驱动的时代,API(应用程序接口)成为了获取数据的重要途径。而通过API代理爬虫,我们可以高效地采集和处理数据,尤其是在面对反爬虫机制时。本文将为你介绍API代理爬虫的基本概念、工作原理以及如何使用。1.什么是API代理爬虫?API代理爬虫是一种结合了......
  • Transformer系列:图文详解Decoder解码器原理
    Encoder-Decoder框架简介理解Transformer的解码器首先要了解Encoder-Decoder框架。在原论文中Transformer用于解决机器翻译任务,机器翻译这种Seq2Seq问题通常以Encoder-Decoder框架来解决,Transformer的网络结构也是基于encoder-decoder框架设计的。这种框架的模型分为两部......
  • 智能加速计算卡设计原理图:628-基于VU3P的双路100G光纤加速计算卡 XCVU3P板卡
    基于VU3P的双路100G光纤加速计算卡 一、板卡概述     基于XilinxUltraScale+16nmVU3P芯片方案基础上研发的一款双口100GFPGA光纤以太网PCI-Expressv3.0x16智能加速计算卡,该智能卡拥有高吞吐量、低延时的网络处理能力以及辅助CPU进行网络功能卸载的能力,达到最......
  • “Datawhale x魔搭 AI夏令营”-AIGC方向-Day1从零入门AI生图原理&实践
    学习内容提要:从通过代码实现AI文生图逐渐进阶,教程偏重图像工作流、微调、图像优化等思路,最后会简单介绍AIGC应用方向、数字人技术(选学)Task01:简单了解一下文生图相关的基础知识具体Datawhale教程学习内容见链接:https://linklearner.com/activity/14/10/24报名赛事链接:https:/......
  • DataWhale-2024夏令营第四期-从零入门AI生图原理&实践-学习笔记
    DataWhale-2024夏令营第四期-从零入门AI生图原理&实践-学习笔记Datawhale(linklearner.com)学习链接AI生图基础知识一、文生图(Text-to-ImageGeneration)历史随着深度学习的发展,近些年来越来越多的AI生图效果通过大语言模型得到了一定的提升。文生图的历史:文生图的概念最......
  • Linux:线程同步机制(互斥锁、读写锁、条件变量、信号量详细分析总结)
    目录速览1、互斥锁(1)What(什么是互斥锁)(2)Why(互斥锁的用途)(3)How(如何使用互斥锁)(4)代码实践2、读写锁(1)What(什么是读写锁)(2)Why(读写锁的作用)(3)How(如何使用读写锁)(4)读写锁的特征3、条件变量(1)What(什么是条件变量)(2)Why(条件变量的作用)(3)How(如何使用条件变量实现线程......