答案:使用ReentrantReadWriteLock可实现读写分离,允许多个读者并发读取,写者独占访问。通过读锁和写锁的配合,保证线程安全;非公平模式下性能高,但可能引发写线程饥饿,启用公平模式可缓解此问题;适用于读多写少场景,如缓存管理,需注意锁降级与避免死锁。
在Java中解决读者写者问题,核心目标是允许多个读者线程同时读取共享资源,但写者线程必须独占访问资源,避免数据不一致。可以通过java.util.concurrent包中的ReentrantReadWriteLock来高效实现。
使用ReentrantReadWriteLock
ReentrantReadWriteLock提供了两个锁:一个读锁和一个写锁。多个线程可以同时持有读锁,但写锁是独占的,且读写不能共存。
以下是典型实现方式:
import java.util.concurrent.locks.ReentrantReadWriteLock;
public class ReadWriteResource {
private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
private final ReentrantReadWriteLock.ReadLock readLock = lock.readLock();
private final ReentrantReadWriteLock.WriteLock writeLock = lock.writeLock();
private String data = "初始数据";
// 读者方法
public void
read() {
readLock.lock();
try {
System.out.println("读取数据: " + data);
Thread.sleep(100); // 模拟读操作耗时
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
} finally {
readLock.unlock();
}
}
// 写者方法
public void write(String newData) {
writeLock.lock();
try {
System.out.println("写入数据: " + newData);
data = newData;
Thread.sleep(200); // 模拟写操作耗时
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
} finally {
writeLock.unlock();
}
}
}
控制读写优先级
默认情况下,ReentrantReadWriteLock是非公平锁,可能导致写线程饥饿(长时间等待)。如果需要更公平的调度,可以在构造时传入true启用公平模式:
private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock(true);
公平模式下,线程按请求顺序获取锁,减少写者饥饿的风险,但吞吐量可能降低。
适用场景与注意事项
这种机制适用于读多写少的场景,例如缓存、配置管理等。
- 读锁支持重入,多个读线程可同时进入。
- 写锁是独占的,且持有写锁的线程也可以获取读锁(锁降级)。
- 不要在持有读锁时尝试获取写锁,会导致死锁。
- 务必在finally块中释放锁,确保异常时也能正确释放。
ReentrantReadWriteLock能简洁有效地解决读者写者问题,兼顾性能与线程安全。

read() {
readLock.lock();
try {
System.out.println("读取数据: " + data);
Thread.sleep(100); // 模拟读操作耗时
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
} finally {
readLock.unlock();
}
}
// 写者方法
public void write(String newData) {
writeLock.lock();
try {
System.out.println("写入数据: " + newData);
data = newData;
Thread.sleep(200); // 模拟写操作耗时
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
} finally {
writeLock.unlock();
}
}
}







