公平锁指线程按请求顺序获取锁,先到先得,避免饥饿。ReentrantLock默认非公平,构造时传true可启用公平模式,如new ReentrantLock(true),确保等待最久的线程优先获得锁。

在Java中,ReentrantLock 可以用来实现公平锁。默认情况下,ReentrantLock是非公平的,但可以通过构造函数参数设置为公平模式。
什么是公平锁?
公平锁是指线程获取锁的顺序严格按照它们请求锁的时间顺序,即先等待的线程优先获得锁(FIFO)。这避免了线程“饥饿”问题,但可能降低吞吐量,因为需要维护排队机制。
使用ReentrantLock实现公平锁
创建 ReentrantLock 时,传入 true 参数即可启用公平策略:
示例代码:
立即学习“Java免费学习笔记(深入)”;
import java.util.concurrent.locks.ReentrantLock; public class FairLockExample { // true 表示使用公平锁 private final ReentrantLock lock = new ReentrantLock(true); public void performTask() { lock.lock(); try { System.out.println(Thread.currentThread().getName() + " 获取了锁"); // 模拟工作 Thread.sleep(100); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } finally { lock.unlock(); } } public static void main(String[] args) { FairLockExample example = new FairLockExample(); Runnable task = example::performTask; // 启动多个线程竞争锁 for (int i = 1; i <= 5; i++) { new Thread(task, "Thread-" + i).start(); } } }
公平锁的注意事项
虽然公平锁更“公正”,但有以下几点需要注意:
- 性能开销较大:公平锁需要维护等待队列,上下文切换更多,吞吐量通常低于非公平锁
- 即使使用公平锁,tryLock() 方法也不会遵循公平性原则
- 某些情况下,唤醒的线程仍需与其他新到达的线程竞争(取决于具体实现和调度)
- 不能完全保证绝对的公平,特别是在高并发或系统调度延迟的情况下
基本上就这些。如果你的应用场景对线程等待时间敏感,比如希望避免个别线程长期得不到执行,可以考虑使用公平锁。否则,默认的非公平锁在大多数情况下表现更好。


