当前位置: 首页 > news >正文

利用aqs构建一个自己的非公平独占锁

功能

具备基本的加锁、解锁功能,并且是非公平的(即允许插队)。

代码

import java.util.concurrent.locks.AbstractQueuedSynchronizer;public class MyNonfairLock {// 内部同步器类private static class Sync extends AbstractQueuedSynchronizer {// 尝试获取锁@Overrideprotected boolean tryAcquire(int acquires) {// 非公平锁直接尝试获取,不考虑队列中是否有等待线程final Thread current = Thread.currentThread();int c = getState();if (c == 0) {// CAS 操作尝试获取锁if (compareAndSetState(0, acquires)) {setExclusiveOwnerThread(current);return true;}}// 可重入支持else if (current == getExclusiveOwnerThread()) {int nextc = c + acquires;if (nextc < 0) // overflowthrow new Error("Maximum lock count exceeded");setState(nextc);return true;}return false;}// 尝试释放锁@Overrideprotected boolean tryRelease(int releases) {int c = getState() - releases;if (Thread.currentThread() != getExclusiveOwnerThread())throw new IllegalMonitorStateException();boolean free = false;if (c == 0) {free = true;setExclusiveOwnerThread(null);}setState(c);return free;}// 是否被当前线程独占@Overrideprotected boolean isHeldExclusively() {return getExclusiveOwnerThread() == Thread.currentThread();}}// 创建同步器实例private final Sync sync = new Sync();// 加锁方法public void lock() {sync.acquire(1);}// 尝试加锁public boolean tryLock() {return sync.tryAcquire(1);}// 解锁方法public void unlock() {sync.release(1);}// 查询当前线程是否持有锁public boolean isHeldByCurrentThread() {return sync.isHeldExclusively();}// 查询锁是否被任何线程持有public boolean isLocked() {return sync.getState() != 0;}
}
  1. 非公平性实现:

在 tryAcquire 方法中,直接尝试获取锁(通过 CAS 操作),不考虑等待队列中是否有其他线程在等待

这与公平锁的区别在于公平锁会先检查队列中是否有等待线程

  1. 可重入支持:

如果当前线程已经是锁的持有者,则增加 state 计数

释放锁时需要释放相同次数

  1. 状态管理:

state = 0 表示锁未被占用

state > 0 表示锁被占用,且数值表示重入次数

  1. 独占模式:

只允许一个线程持有锁

使用 setExclusiveOwnerThread 和 getExclusiveOwnerThread 来跟踪锁的持有者

使用示例

public class MyLockExample {private static final MyNonfairLock lock = new MyNonfairLock();private static int counter = 0;public static void main(String[] args) throws InterruptedException {Runnable task = () -> {for (int i = 0; i < 10000; i++) {lock.lock();try {counter++;} finally {lock.unlock();}}};Thread t1 = new Thread(task);Thread t2 = new Thread(task);t1.start();t2.start();t1.join();t2.join();System.out.println("Final counter value: " + counter); // 应该输出20000}
}
http://www.dtcms.com/a/224325.html

相关文章:

  • 【LUT技术专题】图像自适应3DLUT
  • 设计模式——原型设计模式(创建型)
  • Cypress + React + TypeScript
  • macOS 上安装运行 PowerShell
  • 电路图识图基础知识-常用仪表识图及接线(九)
  • uniapp uni-id Error: Invalid password secret
  • Oracle用户账号过期终极解决方案
  • 嵌入式学习笔记 - STM32 HAL库以及标准库内核以及外设头文件区别问题
  • python 空气质量可视化,数据分析 + 前后端分离 + ppt 演讲大纲
  • 【数据分析】基于Cox模型的R语言实现生存分析与生物标志物风险评估
  • 告别硬编码!用工厂模式优雅构建可扩展的 Spring Boot 应用 [特殊字符]
  • AI炼丹日志-25 - OpenAI 开源的编码助手 Codex 上手指南
  • 修改 vscode 左侧导航栏的文字大小 (更新版)
  • C++文件和流基础
  • uniapp 键盘顶起页面问题
  • 解决8080端口被占问题
  • CppCon 2014 学习:ODB, Advanced Weapons and Tactics
  • 【Python高阶】面向对象
  • VMvare 创建虚拟机 安装CentOS7,配置静态IP地址
  • |从零开始的Pyside2界面编程|绘图、布局及页面切换
  • 2.2HarmonyOS NEXT高性能开发技术:编译优化、内存管理与并发编程实践
  • tomcat服务器以及接受请求参数的方式
  • 尚硅谷redis7 93-97 springboot整合reids之总体概述
  • LLM推理相关指标
  • python分配方案数 2023年信息素养大赛复赛/决赛真题 小学组/初中组 python编程挑战赛 真题详细解析
  • Go 语言的 GC 垃圾回收
  • 核心机制三:连接管理(三次握手)
  • Day08
  • Hbase
  • Web开发实战:HTML+CSS+JS期末复习全梳理