StampedLock如何使用?

本文深入探讨了Java 8中引入的StampedLock,一种比ReadWriteLock更高效的锁机制。StampedLock支持乐观读锁、悲观读锁和写锁,允许在读操作中并发进行写操作,从而提升整体性能。通过具体示例,展示了StampedLock在多线程环境下的使用方法。
  • StampedLock 是从 JDK1.8 开始提供,它的性能比 ReadWriteLock 好
  • StampedLock 支持:乐观读锁、悲观读锁、写锁
  • StampedLock 的悲观读锁、写锁,与 ReadWriteLock 的读锁、写锁用法相似:读读可并行、读写互斥、写写互斥。
  • StampedLock 之所以性能优于 ReadWriteLock,因为它支持乐观读锁。乐观读锁操作,支持一个线程并发进行写操作。
  • StampedLock 不支持重入
  • StampedLock 支持锁的降级和升级
  • StampedLock 可以用悲观读锁调用 readLockInterruptibly() 方法和写锁调用 writeLockInterruptibly() 方法,支持可中断

 

使用示例:

package constxiong.interview;

import java.util.Random;
import java.util.concurrent.locks.StampedLock;

/**
 * 测试 StampedLock
 * @author ConstXiong
 */
public class TestStampedLock {

	private static final StampedLock sl = new StampedLock();
	
	private static volatile int count = 0;
	
	private static final Random r = new Random();
	
	public static void main(String[] args) {
		//启动 5个线程写计数,95 个线程读计数,
		for (int i = 0; i < 100; i++) {
			if (i % 20 == 0) {
				new Thread(() -> {
					try {
						Thread.sleep(r.nextInt(10));
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
					System.out.println(Thread.currentThread().getName() + " 计数新增 1 :" + add());
				}).start();
			} else {
				new Thread(() -> {
					try {
						Thread.sleep(r.nextInt(10));
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
					System.out.println(Thread.currentThread().getName() + " 读计数:" + read());
				}).start();
			}	
		}
	}
	
	/**
	 * 读取计数
	 * @return
	 */
	private static int read() {
		int r;
		long stamp = sl.tryOptimisticRead();
		r = count;
		if (!sl.validate(stamp)) {
			stamp = sl.readLock();
			try {
				r = count;
			} finally {
				sl.unlockRead(stamp);
			}
		}
		return r; 
	}
	
	/**
	 * 计数加 1
	 */
	private static int add() {
		long stamp = sl.writeLock();
		try {
			count++;
		} finally {
			sl.unlockWrite(stamp);
		}
		return count;
	}
	
}

 


【Java面试题与答案】整理推荐

 

### 关于 `StampedLock` 的详细介绍 #### 创建 `StampedLock` 通过如下方式可以创建一个新的 `StampedLock` 对象: ```java StampedLock lock = new StampedLock(); ``` 此操作会初始化一个不持有任何锁的新实例[^3]。 #### 获取不同类型的锁 ##### 乐观读锁 乐观读锁假设在读取期间不会发生修改,因此不需要实际获得锁。如果检测到冲突,则可以选择重试或升级为悲观读锁。 ```java long stamp = lock.tryOptimisticRead(); // 执行读操作... if (!lock.validate(stamp)) { // 如果验证失败, 表明有其他写入者干扰过, 需要重新尝试或者转而使用悲观读锁. } ``` ##### 悲观读锁 当确实存在并发写入的风险时,应该采用悲观的方式锁定资源以防止数据竞争。 ```java long stamp = lock.readLock(); try { // 执行受保护的操作... } finally { lock.unlockRead(stamp); } ``` ##### 写锁 对于需要独占访问权限的情况(即不允许有任何读者),则应申请写锁。 ```java long stamp = lock.writeLock(); try { // 更新状态... } finally { lock.unlockWrite(stamp); } ``` #### 锁转换 有时可能希望先尝试快速路径(如乐观读),然后再根据情况调整策略: ```java long stamp = lock.tryOptimisticRead(); try { if (someCondition()) { // 假设 someCondition() 是决定是否需要更强力保障条件的方法 stamp = lock.convertToReadLock(stamp); // 将乐观读转变为悲观读 try { // 继续执行更严格的逻辑... } finally { lock.unlockRead(stamp); } } } finally { if (lock.isHeldByCurrentThread(stamp)) lock.unlock(stamp); } ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值