16 - ReentrantLock 可重入锁

本文详细介绍了 Java 中的 ReentrantLock,包括其背景、如何保证可见性、可重入特性以及公平锁与非公平锁的概念。ReentrantLock 作为 Lock 接口的一个实现,弥补了 synchronized 的一些不足,提供了中断支持、超时获取锁以及非阻塞获取锁的能力。同时,通过使用 volatile 变量 state 实现可见性保证。此外,文章还讨论了 ReentrantLock 的源码,解析了获取和释放锁的机制以及重入的实现方式。

1. ReentrantLock 概念

1.1 背景

  Java 语言本身提供的 synchronized 也是管程的一种实现,既然 Java 从语言层面已经实现了管程了,那为什么还要在 SDK 里提供另外一种实现呢?难道 Java 标准委员会还能同意“重复造轮子”的方案?很显然它们之间是有巨大区别的。那区别在哪里呢?

  你也许曾经听到过很多这方面的传说,例如在 Java 的 1.5 版本中,synchronized 性能不如 SDK 里面的 Lock,但 1.6 版本之后,synchronized 做了很多优化,将性能追了上来,所以 1.6 之后的版本又有人推荐使用 synchronized 了。那性能是否可以成为“重复造轮子”的理由呢?显然不能。因为性能问题优化一下就可以了,完全没必要“重复造轮子”。

  到这里,关于这个问题,你是否能够想出一条理由来呢?如果你细心的话,也许能想到一点。那就是我们前面文章《05 - 线程死锁了,怎么办?》,提出了一个破坏不可抢占条件方案,但是这个方案 synchronized 没有办法解决。原因是 synchronized 申请资源的时候,如果申请不到,线程直接进入阻塞状态了,而线程进入阻塞状态,啥都干不了,也释放不了线程已经占有的资源。但我们希望的是:

对于“不可抢占”这个条件,占用部分资源的线程进一步申请其他资源时,如果申请不到,可以主动释放它占有的资源,这样不可抢占这个条件就破坏掉了。

如果我们重新设计一把互斥锁去解决这个问题,那该怎么设计呢?我觉得有三种方案:

  1. 能够响应中断。synchronized 的问题是,持有锁 A 后,如果尝试获取锁 B 失败,那么线程就进入阻塞状态,一旦发生死锁,就没有任何机会来唤醒阻塞的线程。但如果阻塞状态的线程能够响应中断信号,也就是说当我们给阻塞的线程发送中断信号的时候,能够唤醒它,那它就有机会释放曾经持有的锁 A。这样就破坏了不可抢占条件了;
  2. 支持超时。如果线程在一段时间之内没有获取到锁,不是进入阻塞状态,而是返回一个错误,那这个线程也有机会释放曾经持有的锁。这样也能破坏不可抢占条件;
  3. 非阻塞地获取锁。如果尝试获取锁失败,并不进入阻塞状态,而是直接返回,那这个线程也有机会释放曾经持有的锁。这样也能破坏不可抢占条件。

  这三种方案可以全面弥补 synchronized 的问题。到这里相信你应该也能理解了,这三个方案就是“重复造轮子”的主要原因,体现在 API 上,就是 Lock 接口的三个方法。详情如下:

/**
* 支持中断的 API
*/
public void lockInterruptibly() throws InterruptedException {
   
   
   
}

/**
* 支持非阻塞的 API
*/
public boolean tryLock() {
   
   
   
}

/**
* 支持超时的 API
*/
public boolean tryLock(long timeout, TimeUnit unit)
       throws InterruptedException {
   
   
   
}

  

1.2 保证可见性

  Java SDK 里面 Lock 的使用,有一个经典的范例,就是try{}finally{},需要重点关注的是在 finally 里面释放锁。这个范例无需多解释,你看一下下面的代码就明白了。但是有一点需要解释一下,那就是可见性是怎么保证的。

  你已经知道 Java 里多线程的可见性是通过 Happens-Before 规则保证的,而 synchronized 之所以能够保证可见性,也是因为有一条 synchronized 相关的规则:synchronized 的解锁 Happens-Before 于后续对这个锁的加锁。那 Java SDK 里面 Lock 靠什么保证可见性呢?例如在下面的代码中,线程 T1 对 value 进行了 +=1 操作,那后续的线程 T2 能够看到 value 的正确结果吗?

class X {
   
   
    private final Lock rtl = new ReentrantLock();
    int value;

    public void addOne() {
   
   
        // 获取锁
        rtl.lock();
        try {
   
   
            value += 1;
        } finally {
   
   
            // 保证锁能释放
            rtl.unlock();
        }
    }
}

  答案必须是肯定的。Java SDK 里面锁的实现非常复杂,下面会详细介绍,这里先简单介绍一下:它是利用了 volatile 相关的 Happens-Before 规则。Java SDK 里面的 ReentrantLock,内部持有一个 volatile 的成员变量 state,获取锁的时候,会读写 state 的值;解锁的时候,也会读写 state 的值(简化后的代码如下面所示)。也就是说,在执行 value+=1 之前,程序先读写了一次 volatile 变量 state,在执行 value+=1 之后,又读写了一次 volatile 变量 state。根据相关的 Happens-Before 规则:

  1. 顺序性规则:对于线程 T1,value+=1 Happens-Before 释放锁的操作 unlock();
  2. volatile 变量规则:由于 state = 1 会先读取 state,所以线程 T1 的 unlock() 操作 Happens-Before 线程 T2 的 lock() 操作;
  3. 传递性规则:线程 T1 的 value+=1 Happens-Before 线程 T2 的 lock() 操作。
class SampleLock {
   
   
    volatile int state;

    // 加锁
    lock() {
   
   
        // 省略代码无数
        state = 1;
    }

    // 解锁
    unlock(
Java并发包(java.util.concurrent)常用类及使用场景 1、locks包中的(如ReentrantLock等),提供更灵活的机制,替代 synchronized。 ● ReentrantLock 可重入,支持公平和非公平,可中断、可限时等待。 ● ReentrantReadWriteLock 读写分离,允许多个读线程同时访问,写线程独占资源。 ● StampedLock (Java 8+)支持乐观读,适用于读多写少场景,性能优于 ReentrantReadWriteLock。 ● Condition 替代 Object.wait()/notify(),实现更精细的线程等待/唤醒机制。 2、atomic包下的原子类:如AtomicInteger、AtomicLong、LongAdder等,原子类提供无线程安全操作,基于CAS(Compare and Swap)实现,适用于计数器、状态标志等场景。 3、提供了线程安全的集合类,支持高并发读写。 ● ConcurrentHashMap:使用节点的思想,即采用“CAS+synchronized”的机制来保证线程安全。 ● CopyOnWriteArrayList/CopyOnWriteArraySet:核心机制是写时复制(Copy-On-Write,简称 COW机制),读操作无,写操作复制新数组,适合读多写少的场景。 ● BlockingQueue(接口):实现类有ArrayBlockingQueue、LinkedBlockingQueue、PriorityBlockingQueue、SynchronousQueue。都是阻塞式队列,支持生产者-消费者模型。 ● ArrayBlockingQueue:基于数组的有界队列。 ● LinkedBlockingQueue:基于链表的无界队列(默认最大为 Integer.MAX_VALUE)。 ● PriorityBlockingQueue:支持优先级的无界队列。 ● SynchronousQueue:无缓冲队列,直接传递任务给消费者。 4、Executor线程池框架,管理线程的生命周期和任务调度。 ExecutorService(接口),实现有ThreadPoolExecutor、ScheduledThreadPoolExecutor,用于执行异步任务,支持定时/延迟任务。 Future/FutureTask,用于获取异步任务执行结果。 5、Executors线程池工具类,内部通过调用ThreadPoolExecutor构造函数来实现,通过该类提供的静态方法可以快速创建一些常用线程池。 ThreadPoolExecutor类是线程池类,可以通过ThreadPoolExecutor类手动创建线程池。 6、Fork/Join框架:是jdk7中新增的并行执行任务的框架,通过 递归任务拆分 和 工作窃取算法,充分利用多核处理器的计算能力。 ForkJoinPool是执行Fork/Join任务的线程池;ForkJoinTask是要执行的任务的基类,常用的有RecursiveAction(无返回值的任务)和RecursiveTask(有返回值的任务)。 7、CompletableFuture实现对多线程进行编排,CompletableFuture提供了非常强大的Future的扩展功能,可以帮助我们简化异步编程的复杂性,提供了函数式编程的能力。 8、AQS(AbstractQueuedSynchronizer抽象排队同步器)是用于构建和同步器的核心框架。几乎所有 Java 并发工具类(如 ReentrantLock、Semaphore、CountDownLatch 、CyclicBarrier 等)的底层实现都依赖于 AQS。它通过一个双向队列(CLH 变体)和状态变量,实现了线程的排队、阻塞与唤醒机制。 9、同步工具类(如Semaphore信号量、CountDownLatch(倒计时门闩)、CyclicBarrier循环屏障、Phaser(阶段器)、Exchanger(交换器)),协调多线程的执行顺序或状态。 ● Semaphore(信号量),用于控制资源访问并发数,可以用在限流等场景。 ● CountDownLatch(倒计时门闩)让主线程等待一组子线程完成任务(一次性)。用于并行任务完成后汇总结果(如多线程加载数据后启动系统)。 ● CyclicBarrier(循环屏障),用于一组线程相互等待,到达屏障后统一执行后续操作(可重复使用),例如分阶段数据处理后合并。 ● Phaser(阶段器):与CyclicBarrier类似,也是一种多线程同步工具,但是支持更灵活的栅栏操作,可以动态地注册和注销参与者,并可以控制各个参与者的到达和离开(复杂版CyclicBarrier)。 ● Exchanger(交换器):两个线程在特定点交换数据,例如生产者-消费者模式中的缓冲区交换。 生成思维导图
05-10
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值