登录
首页 >  文章 >  java教程

ReentrantLock锁详解与使用示例

时间:2025-12-29 23:36:09 460浏览 收藏

对于一个文章开发者来说,牢固扎实的基础是十分重要的,golang学习网就来带大家一点点的掌握基础知识点。今天本篇文章带大家了解《Java ReentrantLock锁控制详解与使用示例》,主要介绍了,希望对大家的知识积累有所帮助,快点收藏起来吧,否则需要时就找不到了!

ReentrantLock是Java中可重入的显式锁,相比synchronized提供更高灵活性,支持公平锁、尝试获取、中断响应等;通过lock()和unlock()手动控制加锁释放,需配合try-finally防止泄漏;具备可重入性,同一线程可多次获取锁,适用于复杂并发场景。

Java如何使用ReentrantLock实现锁控制_Java可重入锁的使用示例与原理解析

Java中,ReentrantLock 是一种可重入的互斥锁,它提供了比 synchronized 更灵活的锁机制。相比 synchronized 的隐式获取和释放,ReentrantLock 需要手动控制加锁与解锁,但带来了更高的可控性,比如支持公平锁、非阻塞尝试获取锁、中断响应等特性。

ReentrantLock 基本使用示例

以下是一个简单的多线程计数器示例,展示如何使用 ReentrantLock 保证线程安全:

import java.util.concurrent.locks.ReentrantLock;

public class Counter {
    private int count = 0;
    private final ReentrantLock lock = new ReentrantLock();

    public void increment() {
        lock.lock();  // 手动加锁
        try {
            count++;
        } finally {
            lock.unlock(); // 必须在finally中释放锁
        }
    }

    public int getCount() {
        lock.lock();
        try {
            return count;
        } finally {
            lock.unlock();
        }
    }
}

说明: 使用 lock() 获取锁,unlock() 释放锁。务必把 unlock() 放在 finally 块中,防止因异常导致锁无法释放。

ReentrantLock 的可重入性

ReentrantLock 支持同一线程多次获取同一把锁,不会造成死锁,这就是“可重入”的含义。每次 lock() 会增加持有计数,对应地需要调用相同次数的 unlock() 才能真正释放锁。

public void methodA() {
    lock.lock();
    try {
        System.out.println("methodA");
        methodB(); // 同一线程再次进入,不会阻塞
    } finally {
        lock.unlock();
    }
}

public void methodB() {
    lock.lock();
    try {
        System.out.println("methodB");
    } finally {
        lock.unlock();
    }
}

上面代码中,同一个线程调用 methodA 后进入 methodB,依然可以成功获取锁,体现了可重入特性。

公平锁与非公平锁

ReentrantLock 构造时可指定是否为公平锁:

  • new ReentrantLock():默认为非公平锁,不保证等待时间最长的线程优先获取锁,性能较高。
  • new ReentrantLock(true):创建公平锁,按请求顺序获取锁,避免线程饥饿,但性能略低。

例如:

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

常用方法与高级功能

ReentrantLock 提供了比 synchronized 更丰富的控制手段:

  • tryLock():尝试获取锁,立即返回 boolean,不会阻塞。
  • tryLock(long timeout, TimeUnit unit):在指定时间内尝试获取锁。
  • lockInterruptibly():可中断地获取锁,适合处理取消操作的场景。
  • isHeldByCurrentThread():判断当前线程是否持有该锁。

示例:带超时的锁获取

if (lock.tryLock(3, TimeUnit.SECONDS)) {
    try {
        // 成功获取锁,执行操作
    } finally {
        lock.unlock();
    }
} else {
    // 获取失败,处理超时逻辑
}

ReentrantLock 与 synchronized 对比

两者都能实现线程同步,但有明显区别:

  • synchronized 是 JVM 层面的内置锁,自动释放;ReentrantLock 是 API 层面的锁,需手动释放。
  • ReentrantLock 支持公平锁、可中断、超时获取等功能,灵活性更高。
  • synchronized 简单易用,适合大多数场景;ReentrantLock 适用于复杂并发控制需求。

基本上就这些。掌握 ReentrantLock 的使用,有助于写出更高效、可控的并发程序。注意合理使用 try-finally 结构,避免死锁或资源泄漏。

到这里,我们也就讲完了《ReentrantLock锁详解与使用示例》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

相关阅读
更多>
最新阅读
更多>
课程推荐
更多>