Unsafe的CAS操作及线程park与unpark

如下是一个参照AQS进行的一个加锁及解锁的简单实现:

  1. 多线程并发进行同步业务操作;
  2. 加锁:尝试进行cas 0->1操作;
  3. 如果加锁成功则进行业务处理,然后进行锁释放 1->0,然后将列头的线程进行唤醒;
  4. 如果加锁失败则先将线程加入到队列中,然后再进行一次CAS加锁尝试及判断;
  5. 在确认此刻锁是被占用状态(cas仍然失败)后方可将当前线程进行park操作;
  6. 线程被唤醒,重新进行cas加锁操作;

需要区别于【无限CAS】判断操作,如果线程过多或大量的耗费cpu资源。park于unpark的运用可以起到阻塞线程跟唤醒线程的功效,可以不会造成cpu资源的大量浪费。


public class Lock {private volatile int status;private static final Unsafe unsafe;// 对于 Java AbstractQueuedSynchronizer (AQS) 中的 stateOffset 字段,// 通常会使用 static final 修饰是因为这个偏移量值在整个类中都是不变的,且所有实例都需要使用相同的偏移量来进行 CAS 操作。// 即:不管有多少个 Lock 对象,对于Bean对象中的status操作的偏移量值是固定的,可以使用 static final// statusOffset 只是一个相对的偏移量,跟status具体的值无关,所以可以多实例共享statusOffsetprivate static final long statusOffset;// 模拟一个先进先出的队列private static final Queue<Thread> threadQueue = new ConcurrentLinkedQueue<>();static {try {// sun.misc 中的class无法像AQS中那样通过 private static final Unsafe unsafe = Unsafe.getUnsafe(); 直接调用Field field = Unsafe.class.getDeclaredField("theUnsafe");field.setAccessible(true);unsafe = (Unsafe) field.get(null);statusOffset = unsafe.objectFieldOffset(Lock.class.getDeclaredField("status"));} catch (Exception e) {throw new RuntimeException(e);}}private void lock() {int expect = 0;int update = 1;while (true) {// 进行cas尝试if (unsafe.compareAndSwapInt(this, statusOffset, expect, update)) {System.out.println(Thread.currentThread().getName() + ":加锁成功");break;} else {// CAS失败,阻塞线程并放入队列threadQueue.offer(Thread.currentThread());// 队头元素Thread peek = threadQueue.peek();// 在进行线程park前,这里一定需要再进行一次cas尝试,否则可能会出现并发情况下一些线程无法被唤醒的可能if(peek == Thread.currentThread()// 如果这里cas失败,那么说明锁肯定还被某个线程占用着没有释放,而当前线程已经确认加入到队列中,// 那么就能确保,解锁逻辑中poll出来的线程肯定有一个是当前线程的,就不会存在线程不会被唤醒的情况了,// 所以else逻辑中当前线程可以尽情park,肯定会有人叫醒它&& unsafe.compareAndSwapInt(this, statusOffset, expect, update)) {// 如果队头元素就是刚刚放进去的元素,并且cas成功了,那么也代表获取锁成功System.out.println(Thread.currentThread().getName() + ":加锁成功");break;} else {try {// 放大被park的延迟,以更好的观测极端情况TimeUnit.MICROSECONDS.sleep(1);} catch (InterruptedException e) {throw new RuntimeException(e);}unsafe.park(false, 0);}}}}protected void unlock() {int expect = 1;int update = 0;unsafe.compareAndSwapInt(this, statusOffset, expect, update);System.out.println(Thread.currentThread().getName() + ":释放锁成功");// 获取队列头并唤醒线程Thread thread = threadQueue.poll();// 这里可能没有拿到,但是 下一刻 队列中就加入了一个新的线程// 需要考虑并发情况下被遗漏的线程,被遗漏的线程会一直得不到唤醒:见lock逻辑中的再次cas操作判断if (thread != null) {unsafe.unpark(thread);}}// 非线程安全的list 被用来作加锁线程安全测试static List<Integer> businessData = new ArrayList<>();public static void business(Integer i) {// CAS成功,执行业务逻辑businessData.add(i);System.out.println(Thread.currentThread().getName() + ":Business logic executed");}public static void main(String[] args) throws InterruptedException {Lock lock = new Lock();// 模拟多个线程并发进行CAS操作for (int i = 0; i < 100; i++) {// 进行线程并发,模拟进行加锁及业务处理int finalI = i;Thread thread = new Thread(() -> {lock.lock();try {business(finalI);} finally {lock.unlock();}});thread.setName("线程:" + (i + 1));thread.start();}// 这里简单进行睡眠,确保线程都已经执行完TimeUnit.SECONDS.sleep(5);System.out.println(businessData.size());System.out.println(businessData);// 逻辑验证一下结果HashSet<Integer> set = new HashSet<>(businessData);for (int i = 0; i < businessData.size(); i++) {if(!set.contains(i)) {throw new RuntimeException("自建锁线程安全测试失败");}}}
}

运行结果如图:

这段代码实现了一个简单的自定义锁 Lock,通过 CAS 操作来实现加锁和解锁的功能,并在加锁失败时将线程放入队列进行阻塞等待。下面是对代码的简要分析:

  1. 在静态代码块中,使用 Unsafe 类获取对 status 字段的偏移量 statusOffset,并初始化了一个空的线程队列 threadQueue

  2. lock() 方法实现了加锁的逻辑:

    • 首先尝试通过 CAS 操作将 status 从 0 改为 1,如果成功表示加锁成功。
    • 如果 CAS 失败,则将当前线程加入队列,并尝试再次通过 CAS 将 status 改为 1,如果成功则表示当前线程获取到了锁。
    • 如果 CAS 失败,说明锁还被其他线程占用,当前线程会被阻塞并进入休眠状态,直到被唤醒后再次尝试获取锁。
  3. unlock() 方法实现了解锁的逻辑:

    • 通过 CAS 操作将 status 从 1 改为 0,表示释放锁成功。
    • 从队列中取出一个线程,并唤醒该线程。
  4. business(Integer i) 方法模拟了业务逻辑,向一个非线程安全的列表中添加元素。

  5. main() 方法模拟了多个线程并发进行 CAS 操作,加锁后执行业务逻辑,最后输出业务数据列表的大小和内容,并验证数据的正确性。

总体来说,这段代码实现了简单的自定义锁,并在多线程环境下进行了加锁和解锁操作,可以保证业务逻辑的线程安全性。但需要注意,这里只是一个简单的示例,实际场景中若涉及更复杂的业务逻辑和线程交互,可能需要更完善的设计和测试。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/769683.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

GDAL中的地理坐标系、投影坐标系及其相互转换

目录 地理坐标系 国内常用地理坐标系 投影坐标系 国内常用投影坐标系&#xff08;不推荐使用&#xff09; 坐标转换 地理坐标转为投影坐标 投影坐标转为地理坐标 地理坐标系 原理参考这篇文章&#xff1a;地理坐标系与投影坐标系区别与联系 https://yunxingluoyun.blog.…

webserver如何从零开始?

我们要做一个项目&#xff0c;过程是怎么样的呢&#xff1f;git clone ...部署&#xff0c;测试&#xff0c;然后开始写么&#xff0c;这样你大概率会“猪脑过载”&#xff0c;对一个项目的每个部分都没有清晰认识&#xff0c;能写出什么来&#xff1f;写之前当然需要测试每个功…

Linux网络协议栈从应用层到内核层③

文章目录 1、write源码剖析2、vfs层进行数据传输3、socket层进行数据传输4、tcp层进行数据传输5、ip层进行数据传输6、网络设备层进行数据传输7、网卡驱动层进行数据传输8、数据传输的整个流程 1、write源码剖析 系统调用原型 ssize_t write(int fildes, const void *buf, si…

Linux 在线yum安装: PostgreSQL 15.6数据库

Linux 在线yum安装&#xff1a; PostgreSQL 15.6数据库 1、PostgreSQL数据库简介2、在线安装PostgreSQL15.63、配置 PostgreSQL的环境变量4、使用默认用户登录PostgreSQL5、配置 PostgreSQL 允许远程登录6、修改 PostgreSQL 默认端口7、创建数据库和表、远程用户zyl8、pgAdmin远…

MATLAB环境下基于离散小波变换和主成分平均的医学图像融合方法

随着计算机技术和生物影像工程的日趋成熟&#xff0c;医学图像为医疗诊断提供的信息越来越丰富。目前&#xff0c;由于医学成像的设备种类繁多&#xff0c;导致医生获得的图像信息差异较大。如何把这些信息进行整合供医生使用成为当务之急。基于此&#xff0c;医学图像融合技术…

Rust 实战练习 - 8. 内存,ASM,外挂 【重磅】

目标&#xff1a; C写一个Demo版本的游戏由浅入深&#xff0c;了解外挂原理Linux/Android下实现内存读取ptrace实现内存修改&#xff08;依赖第三方库&#xff09; 先准备一个C写的小游戏 #include <stdio.h> #include <string.h>struct Role {float pos_x; // …

vue3+vite配置环境变量

1、创建环境变量文件&#xff1a;首先在vue3项目根目录创建.env.development 和 .env.prodution两个文件&#xff0c;分别为开发和生产环境&#xff08;必须.env.开头&#xff0c;需要额外环境&#xff0c;配置自定义的文件名称即可&#xff09; 2、在环境变量文件分别写对应…

Android内存优化项目经验分享 兼顾效率与性能

背景 项目上线一段时间后,回顾重要页面 保证更好用户体验及生产效率&#xff0c;做了内存优化和下载导出优化&#xff0c;具体效果如最后的一节的表格所示。 下面针对拍摄流程的两个页面 预览页 导出页优化实例进行介绍&#xff1a; 一.拍摄前预览页面优化 预览效果问题 存在…

试试前端自动化测试(基础篇)

众所周知的原因&#xff0c;前端作为一种特殊的 GUI 软件&#xff0c;做自动化测试困难重重。在快速迭代&#xff0c;UI 变动大的业务中&#xff0c;自动化测试想要落地更是男上加男 &#x1f436;。 近期的学习过程中&#xff0c;翻阅了众多前端自动化测试相关的文章&#xf…

【兆易创新GD32H759I-EVAL开发板】 关于LVGL 的内存配置

【兆易创新GD32H759I-EVAL开发板】拥有外部32MB的 SDRAM 在使用LVGL时 可以随意分配大小 但是我们也应该明白 所定义的内存大小的 的一些概念 LVGL中 有单独的 定义 LV_MEM_SIZE 定义内存大小 LVLG 中 在定义 显示程序 接口时 还需要用到 lv_disp_draw_buf_init() 分配显存…

MyBatis框架解析与优化

MyBatis 是一个半 ORM&#xff08;对象关系映射&#xff09;框架&#xff0c;它内部封装了 JDBC&#xff0c;开发时只需要关注 SQL 语句本身&#xff0c;不需要花费精力去处理加载驱动、创建连接、创建 statement 等繁杂的过程。 什么是 MyBatis&#xff1f; MyBatis 是一个半…

【3D reconstruction 学习笔记】

三维重建 3D reconstruction 1. 相机几何针孔相机摄像机几何 2. 相机标定线性方程组的解齐次线性方程组的解非线性方程组的最小二乘解透镜相机标定带畸变的相机标定 3. 单视图重建2D平面上的变换3D空间上的变换单视测量无穷远点 无穷远线 无穷远平面影消点 影消线单视重构 4. 三…

天艺制盖邀您参观2024第七届世界燕窝及天然滋补品博览会

2024第七届世界燕窝及天然滋补品博览会 2024年8月7-9日| 上海新国际博览中心 上海燕博会 世界燕窝及天然滋补品展览会暨世界滋补产业生态发展大会&#xff08;简称上海燕博会&#xff09;&#xff0c;2017年创办于中国上海&#xff0c;是一年一度的世界燕窝滋补品行业盛会。…

宁波中墙建材施工过程中,如何确保陶粒复合砌块的垂直度和水平度符合要求?

宁波中墙建材陶粒复合砌块如何使用 确保陶粒复合砌块施工质量的建议&#xff1a; 基层处理&#xff1a;在施工前&#xff0c;确保基层干净、平整、坚固&#xff0c;去除表面的杂物和油污等。 砌块质量&#xff1a;选择质量好、尺寸规格一致的陶粒复合砌块&#xff0c;避免使用有…

【串口开发】android 智能设备开发 知识笔记

1.一般的波特率选择115200,自己玩的可以用9600等随便的 2.为了android方便操作,引入了 implementation com.licheedev:android-serialport:2.1.3包。 不然就得手写了,比如像这样 ,打开串口监听 // 打开串口boolean openSerialPort = mSerialPortManager.setOnOpenSerial…

每天一个数据分析题(二百二十八)

在超参数调参的各种方法中&#xff0c;贝叶斯优化搜索(Bayesian Optimization)是一种非常有效的方法。请问在贝叶斯搜索中&#xff0c;用于估计目标函数并为下一次迭代提供建议的模型是什么&#xff1f; A. 线性回归 B. 随机森林 C. 高斯过程 D. 神经网络 题目来源于CDA模…

vue js有哪些优点和缺点

Vue.js 是一个流行的前端 JavaScript 框架&#xff0c;用于构建用户界面和单页面应用。以下是 Vue.js 的一些主要优点和缺点&#xff1a; 优点&#xff1a; 轻量级和简洁&#xff1a;Vue.js 的核心库专注于视图层&#xff0c;并且非常轻量&#xff0c;这使得它可以很容易地与其…

java算法题每日多道六

138. 随机链表的复制 题目 给你一个长度为 n 的链表&#xff0c;每个节点包含一个额外增加的随机指针 random &#xff0c;该指针可以指向链表中的任何节点或空节点。 构造这个链表的 深拷贝。 深拷贝应该正好由 n 个 全新 节点组成&#xff0c;其中每个新节点的值都设为其对…

JS异步操作

点击按钮触发onScan函数&#xff0c;函数调用扫描二维码这个异步操作后&#xff0c;需要扫描二维码的函数返回结果&#xff0c;可以用Promise来实现。Promise对象状态变为resolved&#xff08;成功&#xff09;或rejected&#xff08;失败&#xff09;&#xff0c;然后将解决&a…

运放PSRR与开关电源纹波分析的实际案例分享!

本文来自看海原创视频教程&#xff1a;《运放秘籍》运算放大器基础精讲及应用第一部*开天 微信公众号&#xff1a;工程师看海 【淘宝】https://m.tb.cn/h.5PAjLi7?tkvmMLW43KO7q CZ3457 「运放秘籍_运算放大器Multisim仿真视频教程第一部开天_工程师看海」 点击链接直接打开 …