李锋镝的博客

  • 首页
  • 时间轴
  • 评论区显眼包🔥
  • 左邻右舍
  • 博友圈
  • 关于我
    • 关于我
    • 另一个网站
    • 我的导航站
    • 网站地图
    • 赞助
  • 留言
  • 🚇开往
Destiny
自是人生长恨水长东
  1. 首页
  2. 原创
  3. 技术
  4. 正文

ReentrantLock深度解析

2025年5月30日 124点热度 0人点赞 0条评论

一、核心设计思想

1. 可重入性(Reentrancy)

定义:同一线程可多次获取同一把锁,通过计数器记录持有次数,释放时需完全解锁(计数器归零)。
示例代码:

ReentrantLock lock = new ReentrantLock();  
lock.lock();  
try {  
    lock.lock(); // 可重入,计数器+1  
    // 临界区代码  
} finally {  
    lock.unlock(); // 计数器-1  
    lock.unlock(); // 计数器归零,释放锁  
}  

2. 公平性(Fairness)

  • 公平锁:按请求顺序(FIFO)分配锁,避免线程饥饿,适合高竞争场景。
  • 非公平锁(默认):允许线程插队,减少上下文切换开销,适合低竞争场景,吞吐量更高。

3. 显式锁控制

  • 手动加锁/解锁:需通过try-finally确保锁释放,避免死锁。
  • 灵活性:支持tryLock(尝试获取)、lockInterruptibly(可中断获取)、超时获取等。

二、底层实现:AQS(AbstractQueuedSynchronizer)

ReentrantLock基于AQS框架实现,通过CLH双向队列管理等待线程,核心依赖state状态变量和节点(Node)机制。

1. AQS核心机制

  • state变量:表示锁的持有计数(0为未锁定,≥1为锁定次数)。
  • CLH队列:保存等待线程的双向链表,每个节点封装线程状态(如CANCELLED、SIGNAL)。

2. 加锁流程(非公平锁为例)

file

3. 解锁流程

public void unlock() {  
    sync.release(1); // 调用AQS的release方法  
}  

// AQS的release方法  
public final boolean release(int arg) {  
    if (tryRelease(arg)) { // 尝试释放锁(ReentrantLock实现)  
        Node h = head;  
        if (h != null && h.waitStatus != 0)  
            unparkSuccessor(h); // 唤醒队列中下一个线程  
        return true;  
    }  
    return false;  
}  

三、公平锁 vs 非公平锁的源码差异

1. 非公平锁(NonfairSync)

final boolean nonfairTryAcquire(int acquires) {  
    Thread current = Thread.currentThread();  
    int c = getState();  
    if (c == 0) {  
        if (compareAndSetState(0, acquires)) { // 直接CAS,允许插队  
            setExclusiveOwnerThread(current);  
            return true;  
        }  
    } else if (current == getExclusiveOwnerThread()) { // 可重入  
        setState(c + acquires);  
        return true;  
    }  
    return false;  
}  

2. 公平锁(FairSync)

protected final boolean tryAcquire(int acquires) {  
    Thread current = Thread.currentThread();  
    int c = getState();  
    if (c == 0) {  
        if (!hasQueuedPredecessors() && // 检查队列是否有等待线程  
            compareAndSetState(0, acquires)) { // 无等待时才CAS  
            setExclusiveOwnerThread(current);  
            return true;  
        }  
    } else if (current == getExclusiveOwnerThread()) { // 可重入  
        setState(c + acquires);  
        return true;  
    }  
    return false;  
}  

四、高级功能与使用场景

1. 可中断锁获取(lockInterruptibly)

ReentrantLock lock = new ReentrantLock();  
try {  
    lock.lockInterruptibly(); // 响应中断  
    // 临界区代码  
} catch (InterruptedException e) {  
    // 处理中断逻辑  
} finally {  
    lock.unlock();  
}  

2. 超时尝试获取锁(tryLock)

if (lock.tryLock(1, TimeUnit.SECONDS)) { // 等待1秒  
    try {  
        // 临界区代码  
    } finally {  
        lock.unlock();  
    }  
} else {  
    // 超时处理  
}  

3. 条件变量(Condition)

对比synchronized的wait/notify:支持多个条件变量,实现精细化线程协作。
示例:生产者-消费者模型

ReentrantLock lock = new ReentrantLock();  
Condition notFull = lock.newCondition(); // 队列未满条件  
Condition notEmpty = lock.newCondition(); // 队列非空条件  

// 生产者  
lock.lock();  
try {  
    while (queue.isFull()) {  
        notFull.await(); // 等待队列未满  
    }  
    queue.add(item);  
    notEmpty.signal(); // 唤醒消费者  
} finally {  
    lock.unlock();  
}  

// 消费者  
lock.lock();  
try {  
    while (queue.isEmpty()) {  
        notEmpty.await(); // 等待队列非空  
    }  
    item = queue.remove();  
    notFull.signal(); // 唤醒生产者  
} finally {  
    lock.unlock();  
}  

五、最佳实践与常见问题

1. 必须手动释放锁

错误示例(可能导致死锁):

lock.lock();  
// 若此处抛出异常,锁无法释放!  
lock.unlock();  

正确做法:

lock.lock();  
try {  
    // 临界区代码  
} finally {  
    lock.unlock(); // 确保锁释放  
}  

2. 避免嵌套锁导致死锁

风险代码:

lockA.lock();  
try {  
    lockB.lock(); // 若另一线程先获取lockB,会死锁  
} finally {  
    lockB.unlock();  
    lockA.unlock();  
}  

解决方案:按固定顺序获取锁(如先锁A后锁B)。

3. 性能调优建议

  • 优先使用非公平锁:默认策略,适合大多数场景。
  • 减少锁粒度:采用分段锁(如ConcurrentHashMap的分段设计)。
  • 监控锁竞争:使用jstack或JFR分析线程阻塞情况。

六、ReentrantLock vs synchronized

特性 ReentrantLock synchronized
实现方式 JDK类,基于AQS JVM内置关键字
锁获取方式 显式调用lock()/unlock() 隐式获取(代码块/方法)
公平性 支持公平/非公平锁 仅非公平锁
可中断性 支持(lockInterruptibly) 不支持
超时机制 支持(tryLock) 不支持
条件变量 支持多个Condition 单一wait/notify
性能 高竞争下更优(可配置策略) Java 6后优化,低竞争下接近
代码复杂度 高(需手动管理锁) 低(自动释放)

七、源码分析:AQS的等待队列

// AQS中的Node类(简化版)  
static final class Node {  
    volatile int waitStatus;      // 等待状态(CANCELLED=1, SIGNAL=-1等)  
    volatile Node prev;           // 前驱节点  
    volatile Node next;           // 后继节点  
    volatile Thread thread;       // 关联的线程  
    Node nextWaiter;              // 条件队列中的下一个节点  
}  
// CLH 队列示意图
Head -> Node(Thread1, SIGNAL) ↔ Node(Thread2, CANCELLED) ↔ Tail

八、总结

设计亮点

  • 基于AQS的模板模式:将锁逻辑委托给子类(公平锁/非公平锁)实现。
  • 分离公平性策略:通过不同Sync子类支持灵活的锁分配机制。
  • 条件变量精细化控制:解决synchronized单一等待集的局限性。

适用场景

  • 需要可中断、超时或公平锁的高并发场景。
  • 复杂线程协作场景(如生产者-消费者模型、线程池任务调度)。
除非注明,否则均为李锋镝的博客原创文章,转载必须以链接形式标明本文链接

本文链接:https://www.lifengdi.com/article/tech/4436

相关文章

  • JAVA之从线程安全说到锁
  • CompletableFuture使用详解
  • 别再背线程池的七大参数了,现在面试官都这么问
  • UUID太长怎么办?快来试试NanoId
  • JAVA关键字之volatile关键字说明
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可
标签: AQS JAVA Lock ReentrantLock 公平锁 可重入锁 锁 非公平锁 高并发
最后更新:2025年5月30日

李锋镝

既然选择了远方,便只顾风雨兼程。

打赏 点赞
< 上一篇
下一篇 >

文章评论

1 2 3 4 5 6 7 8 9 11 12 13 14 15 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 46 47 48 49 50 51 52 53 54 55 57 58 60 61 62 63 64 65 66 67 69 72 74 76 77 78 79 80 81 82 85 86 87 90 92 93 94 95 96 97 98 99
取消回复

寻寻觅觅,冷冷清清,凄凄惨惨戚戚。乍暖还寒时候,最难将息。三杯两盏淡酒,怎敌他、晚来风急!雁过也,正伤心,却是旧时相识。
满地黄花堆积,憔悴损,如今有谁堪摘?守着窗儿,独自怎生得黑!梧桐更兼细雨,到黄昏、点点滴滴。这次第,怎一个愁字了得!

那年今日(01月25日)

  • 1979年:中国左翼文学运动开创者之一郑伯奇逝世
  • 1949年:日本帝国时期的政治家牧野伸显逝世
  • 1924年:第一届奥林匹克冬季运动会在夏蒙尼开幕
  • 1911年:中国第一部专门刑法典颁布
  • 1504年:意大利艺术家米开朗基罗完成大卫雕像
  • 更多历史事件
最新 热点 随机
最新 热点 随机
AI时代,个人技术博客的出路在哪里? 什么是Meta Server? 千万级大表新增字段实战指南:告别锁表与业务中断 在 SQL 中做范围查询时,使用 BETWEEN AND 和直接用 >/=/ 深度解析 Disruptor:无锁队列的高性能实现与实践 精通Linux根目录:核心文件夹深度解析与实战指南
玩博客的人是不是越来越少了?准备入手个亚太的ECS,友友们有什么建议吗?AI时代,个人技术博客的出路在哪里?使用WireGuard在Ubuntu 24.04系统搭建VPNWordPress实现用户评论等级排行榜插件WordPress网站换了个字体,差点儿把样式换崩了
基于Java8的Either类 居家办公了~ 祝大家六一儿童节快乐~~~ IntelliJ IDEA 2020.3.x永久白嫖(Windows/Mac) 看病难~取药难~~ 睡觉睡不踏实
标签聚合
WordPress K8s 分布式 AI编程 多线程 设计模式 JAVA ElasticSearch Redis SpringBoot JVM SQL AI docker IDEA 架构 MySQL 日常 数据库 Spring
友情链接
  • Blogs·CN
  • Honesty
  • Mr.Sun的博客
  • 临窗旋墨
  • 哥斯拉
  • 彬红茶日记
  • 志文工作室
  • 懋和道人
  • 搬砖日记
  • 旧时繁华
  • 林羽凡
  • 瓦匠个人小站
  • 皮皮社
  • 知向前端
  • 蜗牛工作室
  • 韩小韩博客
  • 风渡言

COPYRIGHT © 2026 lifengdi.com. ALL RIGHTS RESERVED.

域名年龄

Theme Kratos Made By Dylan

津ICP备2024022503号-3

京公网安备11011502039375号