当前位置: 首页 > 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}
}

相关文章:

  • 【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界面编程|绘图、布局及页面切换
  • 哪些网站可以做帮助文档/合肥网络推广
  • 打电话推销做网站的是真的吗/百度注册网站
  • 如何做采集网站/企业营销策划案例
  • 免费网站免费无遮挡/app广告联盟平台
  • wordpress 自动采集插件/成都seo优化公司
  • 坪山附近公司做网站建设多少钱/腾讯云域名购买