StampedLock类是一种高性能的读写锁,它通过引入乐观读和写锁的优化机制,提高了多线程环境下的并发性能,他支持三种访问模式:悲观读、写和乐观读,可以根据不同的业务场景选择适合的锁策略,相比传统的读写锁,StampedLock能够更好地利用多核处理器的优势,减少线程间的竞争和阻塞,从而提升系统的吞吐量和响应速度。
官方文档地址:https://docx.iamqiang.com/jdk11/api/java.base/java/util/concurrent/locks/StampedLock.html
StampedLock是一个优化的读写锁,它在多核处理器上提供了比ReentrantReadWriteLock更高的性能,与传统的读写锁不同,StampedLock支持三种访问模式:读、写和乐观读,并且这三种模式都可以相互转换。
假设有一个在线书店系统,其中一个关键功能是书籍的库存更新,每当用户购买书籍时,系统需要从库存中减去相应的数量,同时,为了提供良好的用户体验,系统还需要实时显示每本书的当前库存量,以供其他用户参考。在这个场景中,库存更新操作(写操作)和库存查询操作(读操作)是频繁发生的,而且,多个用户可能同时查询同一本书的库存,但同一时间只有一个用户能够更新库存。
可以使用StampedLock解决这个问题,如下操作:
StampedLock 类中的 asReadLock() 方法用于获取一个 Lock 视图,该视图具有与 StampedLock 的读锁相同的锁定含义,可以使用返回的 Lock 对象进行读锁定,就像使用 ReentrantReadWriteLock 的读锁一样,但是,通常建议使用 StampedLock 的其他方法来获取读锁,因为它们可以提供更精细的控制和更高的性能。
下面是一个简单的例子,演示了使用 StampedLock 类的基本使用方法,这个例子创建了一个简单的计数器类,该类使用 StampedLock 来同步对内部计数器的访问,如下代码:
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.StampedLock;
/**
* @创建人 程序员古德 <br>
* @创建时间 2024/1/18 12:00 <br>
* @修改人 暂无 <br>
* @修改时间 暂无 <br>
* @版本历史 暂无 <br>
*/
public class Counter {
private int count;
private final StampedLock stampedLock = new StampedLock();
// 使用 StampedLock 的 asReadLock() 方法获取读锁
public void readCountWithLock() {
Lock readLock = stampedLock.asReadLock();
readLock.lock(); // 获取读锁
try {
System.out.println("Current count: " + count);
} finally {
readLock.unlock(); // 释放读锁
}
}
// 使用 StampedLock 的普通读方法
public int readCountWithStamp() {
long stamp = stampedLock.tryOptimisticRead(); // 尝试乐观读
int currentCount = count;
// 检查乐观读后数据是否被修改
if (!stampedLock.validate(stamp)) {
// 如果数据被修改,获取读锁重新读取
stamp = stampedLock.readLock();
try {
currentCount = count;
} finally {
stampedLock.unlockRead(stamp);
}
}
return currentCount;
}
// 增加计数器的值
public void incrementCount() {
long stamp = stampedLock.writeLock(); // 获取写锁
try {
count++;
} finally {
stampedLock.unlockWrite(stamp); // 释放写锁
}
}
public static void main(String[] args) throws InterruptedException {
Counter counter = new Counter();
// 启动一个线程来增加计数器的值
Thread incrementThread = new Thread(() -> {
for (int i = 0; i < 5; i++) {
counter.incrementCount();
try {
Thread.sleep(100); // 休眠以模拟工作负载
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
// 启动一个线程来读取计数器的值(使用 Lock)
Thread readThreadWithLock = new Thread(() -> {
for (int i = 0; i < 5; i++) {
counter.readCountWithLock();
try {
Thread.sleep(100); // 休眠以模拟工作负载
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
// 启动一个线程来读取计数器的值(使用 stamp)
Thread readThreadWithStamp = new Thread(() -> {
for (int i = 0; i < 5; i++) {
System.out.println("Current count (stamp): " + counter.readCountWithStamp());
try {
Thread.sleep(100); // 休眠以模拟工作负载
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
// 启动所有线程
incrementThread.start();
readThreadWithLock.start();
readThreadWithStamp.start();
// 等待所有线程完成
incrementThread.join();
readThreadWithLock.join();
readThreadWithStamp.join();
}
}
在上面代码中,Counter 类有一个 count 变量,它可以通过 incrementCount 方法来增加,读取计数器值的方法有两种:readCountWithLock 使用 asReadLock() 方法返回的 Lock 对象进行同步,而 readCountWithStamp 则使用 StampedLock 的乐观读和读锁功能。在 main 方法中,启动了三个线程,一个用于增加计数器的值,另外两个用于读取计数器的值(一个使用 Lock,另一个使用 stamp)。
asReadLock() 方法提供了普通 Lock 的方式,但通常建议直接使用 StampedLock 的其他方法(如 tryOptimisticRead、readLock、unlockRead 等),因为它们提供了更高级别的并发控制和性能优化。
StampedLock提供了一种高效的线程同步方式,与传统的读写锁相比,如:ReentrantReadWriteLock,StampedLock则在某些方面展现出了其独特的优势,如下分析:
优点:
缺点:
使用建议:
StampedLock和ReentrantReadWriteLock都是Java中用于同步的机制,它们允许多个线程同时读取共享资源,但在写入时要求独占访问,尽管它们的目的相似,但在设计、性能和适用场景上存在一些关键区别:
在设计上:
在性能上:
在适用场景上:
其他对比:
END!