ReentrantLock提供比synchronized更灵活的线程同步控制,支持手动加锁释放、尝试获取锁、定时等待、可中断及公平锁机制,并可通过Condition实现多条件等待通知,适用于复杂并发场景。

Java中的ReentrantLock是java.util.concurrent.locks包下的一个可重入互斥锁,相比synchronized关键字,它提供了更灵活的线程同步控制方式。通过手动加锁和释放锁,开发者可以实现更复杂的同步逻辑,比如尝试获取锁、定时等待锁、可中断锁等。下面详细介绍如何在实际开发中使用ReentrantLock进行线程同步。
使用ReentrantLock的第一步是创建其实例。通常建议将锁声明为private final字段,以确保线程安全和不可变性。
基本操作流程如下:
示例如下:
立即学习“Java免费学习笔记(深入)”;
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(); // 释放锁
}
}
public int getCount() {
lock.lock();
try {
return count;
} finally {
lock.unlock();
}
}
}
注意:unlock()必须放在finally块中,确保即使发生异常也能释放锁,避免死锁。
ReentrantLock支持非阻塞或限时等待获取锁,这在避免死锁或提升响应性方面非常有用。
tryLock()方法有两种形式:
示例:限制等待锁的时间
public boolean timedIncrement() {
boolean acquired = false;
try {
acquired = lock.tryLock(1, TimeUnit.SECONDS);
if (acquired) {
count++;
return true;
} else {
System.out.println("未能在1秒内获取锁");
return false;
}
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
return false;
} finally {
if (acquired) {
lock.unlock();
}
}
}
这种方式适合对响应时间敏感的场景,比如高并发服务接口。
ReentrantLock支持构造公平锁。默认情况下是非公平锁,即线程抢占式获取锁,效率高但可能造成某些线程长期等待。
如果希望线程按请求顺序获取锁,可在构造时传入true:
private final ReentrantLock fairLock = new ReentrantLock(true); // 公平锁
公平锁能减少线程饥饿问题,但性能开销略大,因为需要维护等待队列。应根据实际场景权衡选择。
ReentrantLock配合Condition接口可替代Object的wait/notify机制,实现更精细的线程通信。
Condition允许创建多个等待条件,每个Condition对应一个等待队列。
示例:生产者-消费者模型
import java.util.LinkedList;
import java.util.Queue;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;
public class BoundedQueue<T> {
private final Queue<T> queue = new LinkedList<>();
private final int maxSize;
private final ReentrantLock lock = new ReentrantLock();
private final Condition notFull = lock.newCondition();
private final Condition notEmpty = lock.newCondition();
public BoundedQueue(int size) {
this.maxSize = size;
}
public void put(T item) throws InterruptedException {
lock.lock();
try {
while (queue.size() == maxSize) {
notFull.await(); // 等待队列不满
}
queue.offer(item);
notEmpty.signal(); // 通知消费者
} finally {
lock.unlock();
}
}
public T take() throws InterruptedException {
lock.lock();
try {
while (queue.isEmpty()) {
notEmpty.await(); // 等待队列不空
}
T item = queue.poll();
notFull.signal(); // 通知生产者
return item;
} finally {
lock.unlock();
}
}
}
使用Condition的好处是可以针对不同条件独立等待和唤醒,比synchronized更灵活。
基本上就这些。ReentrantLock提供了比synchronized更强大的功能,但也要求开发者更加小心地管理锁的获取与释放。只要遵循“加锁后务必释放”的原则,并合理利用tryLock和Condition,就能写出高效且安全的并发程序。
以上就是在Java中如何使用ReentrantLock实现线程同步_ReentrantLock操作实践指南的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号