ReentrantLock 介绍-Java

ReentrantLock 是 Java 并发包(java.util.concurrent.locks)提供的一种显式锁机制。相比于 synchronizedReentrantLock 提供了更多的锁控制功能,例如可中断锁、定时锁、条件变量和公平锁。

1. ReentrantLock

ReentrantLock 是一种 可重入的显式锁。它的可重入性允许同一个线程多次获取同一把锁(无需陷入死锁),并通过手动调用 lock()unlock() 实现锁的获取和释放。

  • 显式锁:需要手动获取(lock())和释放(unlock())。
  • 可重入性:同一线程可以多次获取同一把锁,每次获取都需显式释放。

2. ReentrantLock 的基本用法

以下是 ReentrantLock 的基本用法:

  1. 创建锁对象:
  2. 获取锁:
  3. 释放锁:

示例代码:基本用法

import java.util.concurrent.locks.ReentrantLock;

public class ReentrantLockExample {
    private final ReentrantLock lock = new ReentrantLock();
    private int counter = 0;

    public void increment() {
        lock.lock(); // 获取锁
        try {
            counter++;
        } finally {
            lock.unlock(); // 确保释放锁
        }
    }

    public int getCounter() {
        lock.lock();
        try {
            return counter;
        } finally {
            lock.unlock();
        }
    }
}

3. ReentrantLock 的高级功能

ReentrantLock 提供了以下增强功能,相较于 synchronized 更加灵活。

(1) 可中断锁

线程在尝试获取锁时可以被中断。这是通过 lockInterruptibly() 方法实现的。

lock.lockInterruptibly(); // 可中断的获取锁

使用示例

public void safeMethod() {
    try {
        lock.lockInterruptibly(); // 如果被中断,将抛出 InterruptedException
        // 关键逻辑
    } catch (InterruptedException e) {
        System.out.println("Thread was interrupted while waiting for lock");
    } finally {
        lock.unlock();
    }
}

(2) 锁超时

支持尝试在指定时间内获取锁,如果超时仍未获取到锁,则返回 false

if (lock.tryLock(2, TimeUnit.SECONDS)) {
    try {
        // 获取锁后执行逻辑
    } finally {
        lock.unlock();
    }
} else {
    System.out.println("Failed to acquire lock within 2 seconds");
}

(3) 公平锁与非公平锁

  • 非公平锁(默认):线程获取锁的顺序不一定按照请求锁的顺序,性能更高。
  • 公平锁:线程获取锁时按照请求锁的先后顺序,保证公平性。

创建公平锁

ReentrantLock lock = new ReentrantLock(true); // 公平锁

(4) 条件变量

ReentrantLock 提供了 Condition 对象,用于线程之间的协调。一个锁可以有多个条件变量,通过条件变量可以实现更灵活的线程同步。

常用方法

  • await():使线程等待,并释放锁。
  • signal():唤醒一个等待线程。
  • signalAll():唤醒所有等待线程。

示例:生产者-消费者模型

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;

public class ProducerConsumer {
    private final ReentrantLock lock = new ReentrantLock();
    private final Condition notEmpty = lock.newCondition();
    private final Condition notFull = lock.newCondition();

    private final int[] buffer = new int[10];
    private int count = 0, putIndex = 0, takeIndex = 0;

    public void produce(int value) throws InterruptedException {
        lock.lock();
        try {
            while (count == buffer.length) {
                notFull.await(); // 等待缓冲区有空位
            }
            buffer[putIndex] = value;
            putIndex = (putIndex + 1) % buffer.length;
            count++;
            notEmpty.signal(); // 唤醒消费者
        } finally {
            lock.unlock();
        }
    }

    public int consume() throws InterruptedException {
        lock.lock();
        try {
            while (count == 0) {
                notEmpty.await(); // 等待缓冲区有数据
            }
            int value = buffer[takeIndex];
            takeIndex = (takeIndex + 1) % buffer.length;
            count--;
            notFull.signal(); // 唤醒生产者
            return value;
        } finally {
            lock.unlock();
        }
    }
}

4. ReentrantLock 的实现原理

(1) 基于 AQS 实现

ReentrantLock 的核心是基于 AbstractQueuedSynchronizer (AQS) 实现的,它维护了一个 FIFO 的线程等待队列。

  • 独占模式ReentrantLock 的锁是一种独占锁,只有一个线程可以持有。
  • 可重入性ReentrantLock 记录当前线程获取锁的次数(通过 state 变量),每次释放锁时 state 减少,直到为 0 时完全释放。

(2) 加锁与解锁

  • lock() 方法通过 CAS 操作尝试将状态从 0 修改为 1
  • 如果锁已经被占用,线程会加入 AQS 的等待队列。
  • 释放锁时,unlock() 会将状态从 1 修改为 0,并唤醒等待队列中的线程。

5. ReentrantLock 的优缺点

优点

  1. 灵活性更高:支持可中断锁。支持超时获取锁。支持条件变量,灵活实现线程协调。
  2. 公平锁:提供公平锁选项,保证线程按请求顺序获取锁。
  3. 更适合高并发场景:性能更高,适用于复杂并发需求。

缺点

  1. 复杂性增加:必须显式调用 lock() 和 unlock(),使用不当可能导致死锁。
  2. 代码冗长:相较于 synchronized,代码更加冗长。

6. 使用建议

使用 ReentrantLock 的场景

  • 需要使用 超时锁中断锁
  • 需要实现复杂的线程同步逻辑(如条件变量)。
  • 需要公平锁机制确保线程按顺序执行。
  • 需要更高性能或灵活性时。

使用 synchronized 的场景

  • 同步逻辑简单,代码清晰是首要需求。
  • 不需要复杂的线程协调功能。
Java碎碎念 文章被收录于专栏

来一杯咖啡,聊聊Java的碎碎念呀

全部评论

相关推荐

行云流水1971:这份实习简历的优化建议: 结构清晰化:拆分 “校园经历”“实习经历” 板块(当前内容混杂),按 “实习→校园→技能” 逻辑排版,求职意向明确为具体岗位(如 “市场 / 运营实习生”)。 经历具象化:现有描述偏流程,需补充 “动作 + 数据”,比如校园活动 “负责宣传” 可加 “运营公众号发布 5 篇推文,阅读量超 2000+,带动 300 + 人参与”;实习内容补充 “协助完成 XX 任务,效率提升 X%”。 岗位匹配度:锚定目标岗位能力,比如申请运营岗,突出 “内容编辑、活动执行” 相关动作;申请市场岗,强化 “资源对接、数据统计” 细节。 信息精简:删减冗余表述(如重复的 “负责”),用短句分点,比如 “策划校园招聘会:联系 10 + 企业,组织 200 + 学生参与,到场率达 85%”。 技能落地:将 “Office、PS” 绑定经历,比如 “用 Excel 整理活动数据,输出 3 份分析表;用 PS 设计 2 张活动海报”,避免技能单独罗列。 优化后需强化 “经历 - 能力 - 岗位需求” 的关联,让实习 / 校园经历的价值更直观。 若需要进一步优化服务,私信
实习,投递多份简历没人回...
点赞 评论 收藏
分享
评论
2
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务