StampedLock通过戳记机制提升读写性能,支持乐观读、悲观读和写锁三种模式,适用于读多写少场景。

在高并发场景下,读操作通常远多于写操作。传统的 ReentrantReadWriteLock 虽然支持读写分离,但在读线程非常多时容易导致写线程“饥饿”。Java 8 引入的 StampedLock 提供了一种更高效的读写锁机制,通过使用一种叫“戳记(stamp)”的机制来提升性能。
与传统锁不同,Stamped7Lock 不基于 AQS 实现,且不直接实现 Lock 接口。它的主要优势在于:
StampedLock 支持三种模式:写锁、悲观读锁、乐观读。下面分别说明用法。
1. 写锁(Write Lock)写锁是独占的,获取时会阻塞所有其他读写操作。
立即学习“Java免费学习笔记(深入)”;
private final StampedLock lock = new StampedLock();
public void writeData(int value) {
long stamp = lock.writeLock(); // 阻塞直到获得写锁
try {
this.data = value;
} finally {
lock.unlockWrite(stamp); // 必须用对应 stamp 解锁
}
}
类似传统读锁,允许多个读线程同时访问,但会被写锁阻塞。
public int readData() {
long stamp = lock.readLock();
try {
return this.data;
} finally {
lock.unlockRead(stamp);
}
}
这是 StampedLock 最大的亮点。它假设读期间没有写操作,先获取 stamp 并读取数据,最后验证 stamp 是否有效。
public int optimisticReadData() {
long stamp = lock.tryOptimisticRead(); // 非阻塞,立即返回 stamp
int value = this.data; // 读取共享变量
if (!lock.validate(stamp)) { // 检查期间是否有写操作
// 有写操作发生,转为悲观读
stamp = lock.readLock();
try {
value = this.data;
} finally {
lock.unlockRead(stamp);
}
}
return value;
}
StampedLock 不支持直接将读锁升级为写锁(会死锁),但可以通过检查和重试机制实现安全升级。
public void updateIfEqual(int expected, int newValue) {
long stamp = lock.readLock();
try {
while (this.data == expected) {
// 尝试升级为写锁
long ws = lock.tryConvertToWriteLock(stamp);
if (ws != 0L) { // 升级成功
stamp = ws;
this.data = newValue;
break;
} else {
// 升级失败,释放读锁并重新获取写锁
lock.unlockRead(stamp);
stamp = lock.writeLock();
}
}
} finally {
lock.unlock(stamp); // 统一释放(兼容读/写)
}
}
基本上就这些。StampedLock 在读多写少的场景中表现优异,特别是适合缓存、配置管理等高频读取的模块。只要注意正确使用 stamp 和避免错误的锁升级逻辑,就能显著提升并发效率。
以上就是Java中如何使用StampedLock提高读写效率的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号