16.ABA问题

文章目录

  • ABA问题
    • 1.什么是ABA问题?
    • 2.ABA问题解决方案
      • 2.1.使用AtomicStampedReference解决ABA问题
      • 2.2.使用AtomicMarkableReference解决ABA问题

ABA问题

因为CAS操作的原子性能高,在JUC中广泛被应用,但是如果使用的不合理,CAS操作就会存在ABA问题!

1.什么是ABA问题?

ABA问题是指在并发编程中,由于CAS(比较并交换)操作的特性,可能出现的一种问题。CAS是一种乐观锁机制,用于实现多线程环境下的原子操作。它通常包含三个参数:期望值更新值当前值。CAS操作会比较当前值和期望值,如果相等,则将当前值更新为新值,否则不进行任何操作。

CAS操作的局限性导致了ABA问题的出现。具体而言,ABA问题通常在以下情况下出现:

  1. 并发场景下的共享变量修改:多个线程同时对一个共享变量进行修改和操作。
  2. CAS操作的原子性:CAS操作只能检查共享变量当前的值是否和预期值相等,并在相等时才进行更新操作。
  3. 中间值的变化:如果在CAS操作执行过程中,共享变量的值经历了从预期值A到其他值B,再回到A的变化,那么CAS操作可能会错误地认为共享变量的值没有发生变化。

举例来说,假设一个共享变量的初始值为A,在某个线程执行CAS操作之前,另一个线程将该变量的值从A修改为B,然后再将其修改回A。如果执行CAS操作的线程只关注共享变量的当前值是否和预期值相等,而不考虑中间值的变化,那么CAS操作会错误地认为共享变量的值没有发生变化,导致出现问题。

因此,ABA问题的产生是由于CAS操作只关注共享变量的当前值,而忽略了中间值的变化。这种问题可能会导致并发控制失效,数据不一致等后果,因此需要针对性地设计解决方案来解决ABA问题。

下面通过一个案例,来了解一下ABA问题

假设现在有一个栈,该栈是采用单链表进行实现的,元素的插入和删除都发生在单链表的头部。

假设,线程1 和线程2 是两在堆栈上并发操作的线程,其中线程1计划从head位置通过CAS操作进行元素“B”的弹出操作

在这里插入图片描述

此时线程1刚好启动CAS执行,但是还没有开始执行,线程2抢在线程1前面弹出了元素“B”,并且压入了一个新的元素“C”,再压入了“B”,此时通过CAS操作后 head位置还是“B”

这个时候切换到线程1执行,通过CAS发现线程2的位置还是 “B”操作成功!将“B”弹出并压入新的元素“B”,尽管线程1操作成功,但是这样会存在一个很大问题

在这里插入图片描述

1.已知 栈顶的元素为 B,这个时候线程A知道 B.next = NULL , 希望使用CAS操作,CAS(B,A)将栈顶的元素替换成A,从而将B元素从堆栈中弹出

在这里插入图片描述

2.但是由于线程1 和线程2是并发执行的,CPU时间片不巧分配给了线程2,线程2弹出了B,A元素,然后压入了C,B,最终线程又将head的位置变成了B

在这里插入图片描述

3.接下来线程1从新获得CPU时间片,执行CAS(B,A),操作成功,将B弹出压入了但是此时,链表就断开了,C元素不在存储于堆栈中了,被丢掉了!

在这里插入图片描述

这就是ABA引发的不正常的状态

2.ABA问题解决方案

很多乐观锁的实现版本都是使用版本号(version)的方式来解决ABA问题。每次对共享变量进行修改时,版本号都会相应地递增。这样,在执行CAS操作时,除了比较共享变量的值外,还会比较版本号,从而更精确地判断共享变量是否发生了变化。

2.1.使用AtomicStampedReference解决ABA问题

AtomicStampedReference 是 Java 并发包中提供的一个工具类,用于解决 CAS 操作中的 ABA 问题。与 AtomicReference 不同,AtomicStampedReference 在每个引用值的基础上都附加了一个整型的标记(stamp),用来记录引用值的版本号。这个版本号在每次更新引用值时会自动增加,因此可以用来检测引用值是否发生过变化,从而避免 ABA 问题。

具体来说,AtomicStampedReference 的主要特性包括:

  1. 原子性操作: AtomicStampedReference 提供了一系列原子性的操作方法,包括 compareAndSetgetset 等,确保多线程环境下的安全访问。
  2. 版本号标记: 每个引用值都与一个整型的版本号相关联,用于标记引用值的版本信息。版本号会在每次更新引用值时自动增加,从而在CAS操作中额外检测引用值的变化。
  3. 解决ABA问题: 通过版本号标记,AtomicStampedReference 可以检测到引用值的变化过程,即使引用值发生了从 A 到 B 再到 A 的变化,只要版本号不同,CAS操作就会失败,从而避免了ABA问题。

使用 AtomicStampedReference 解决ABA问题的一般步骤如下:

  1. 初始化 AtomicStampedReference,并指定初始的引用值和版本号。
  2. 在每次对共享变量进行修改时,使用 AtomicStampedReference 的原子性方法,同时更新引用值和版本号。
  3. 在CAS操作中,除了比较共享变量的值外,还要比较版本号是否与预期版本号相同。只有在共享变量的值和版本号都与预期值相同时,才执行更新操作。

通过这种方式,AtomicStampedReference 可以有效地解决CAS操作中的ABA问题,提高了并发环境下的安全性和可靠性。

    @Test@DisplayName("使用AtomicStampedReference解决ABA问题")public void testAtomicStampedReference() throws InterruptedException {// 初始化 AtomicStampedReference,初始值为初始对象和版本号为0AtomicStampedReference<String> atomicStampedRef = new AtomicStampedReference<>("初始值", 0);// 创建两个线程,分别进行CAS操作Thread thread1 = new Thread(() -> {// 获取当前版本号int stamp = atomicStampedRef.getStamp();// 获取当前引用值String value = atomicStampedRef.getReference();log.info("线程1开始执行,当前版本号:{},当前值:{}", stamp, value);// 模拟线程1执行耗时操作try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}// 尝试更新引用值和版本号boolean success = atomicStampedRef.compareAndSet(value, "新值1", stamp, stamp + 1);if (success) {log.info("线程1更新成功,新版本号:{}", stamp + 1);} else {log.error("线程1更新失败,当前值已经被其他线程修改");}});Thread thread2 = new Thread(() -> {// 获取当前版本号int stamp = atomicStampedRef.getStamp();// 获取当前引用值String value = atomicStampedRef.getReference();log.info("线程2开始执行,当前版本号:{},当前值:{}", stamp, value);// 尝试更新引用值和版本号boolean success = atomicStampedRef.compareAndSet(value, "新值2", stamp, stamp + 1);if (success) {log.info("线程2更新成功,新版本号:{}", stamp + 1);} else {log.error("线程2更新失败,当前值已经被其他线程修改");}});// 启动线程thread1.start();thread2.start();// 等待线程执行完成thread1.join();thread2.join();// 输出最终结果log.info("最终值:{}", atomicStampedRef.getReference());}

在这里插入图片描述

2.2.使用AtomicMarkableReference解决ABA问题

AtomicMarkableReference 是 AtomicStampedReference 简化版本,他不关心修改过几次,他只关心有没有被修改过。

  1. AtomicMarkableReference 是 Java 中提供的一种原子引用类,用于解决 CAS 操作中的 ABA 问题。与 AtomicStampedReference 类似,AtomicMarkableReference 也使用了版本号(mark)的方式来确保 CAS 操作的正确性。

    特性:

    1. AtomicMarkableReference 内部包含了一个引用对象和一个标记(mark),它们共同构成了一个原子更新单元。
    2. 标记(mark)是一个布尔值,用于辅助 CAS 操作,标记位的变化表明了引用对象的状态变化。
    3. 提供了 compareAndSet 方法来进行原子更新,确保引用对象和标记的更新是原子性的。
    4. 使用 AtomicMarkableReference 可以避免 CAS 操作中的 ABA 问题,从而更安全地进行并发编程。

    使用过程:

    1. 创建 AtomicMarkableReference 对象时,需要传入初始引用对象和初始标记。
    2. 使用 get 方法可以获取当前的引用对象和标记。
    3. 使用 compareAndSet 方法进行原子更新,需要传入旧的引用对象、新的引用对象、旧的标记和新的标记,只有在旧的引用对象和标记与当前值相同时,才会更新为新的值。
    4. 提供了一种线程安全的机制,可以避免 CAS 操作中的 ABA 问题,确保并发操作的安全性。
    @Test@DisplayName("测试使用AtomicMarkableReference解决ABA问题")public void testAtomicMarkableReference() throws InterruptedException {// 初始化 AtomicMarkableReference,初始值为初始对象和标记为falseAtomicMarkableReference<String> atomicMarkableRef = new AtomicMarkableReference<>("初始值", false);// 创建两个线程,分别进行CAS操作Thread thread1 = new Thread(() -> {// 获取当前标记boolean[] markHolder = new boolean[1];// 获取当前引用值String value = atomicMarkableRef.get(markHolder);log.info("线程1开始执行,当前标记:{},当前值:{}", markHolder[0], value);// 模拟线程1执行耗时操作try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}// 尝试更新引用值和标记boolean success = atomicMarkableRef.compareAndSet(value, "新值1", markHolder[0], true);if (success) {log.info("线程1更新成功,新标记:{}", true);} else {log.error("线程1更新失败,当前值已经被其他线程修改");}});Thread thread2 = new Thread(() -> {// 获取当前标记boolean[] markHolder = new boolean[1];// 获取当前引用值String value = atomicMarkableRef.get(markHolder);log.info("线程2开始执行,当前标记:{},当前值:{}", markHolder[0], value);// 尝试更新引用值和标记boolean success = atomicMarkableRef.compareAndSet(value, "新值2", markHolder[0], true);if (success) {log.info("线程2更新成功,新标记:{}", true);} else {log.error("线程2更新失败,当前值已经被其他线程修改");}});// 启动线程thread1.start();thread2.start();// 等待线程执行完成thread1.join();thread2.join();// 输出最终结果log.info("最终值:{}", atomicMarkableRef.getReference());}

在这里插入图片描述

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

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

相关文章

算法 - hash表 - 2244. 完成所有任务需要的最少轮数 思路题解

2244. 完成所有任务需要的最少轮数 文章目录 [2244. 完成所有任务需要的最少轮数](https://leetcode.cn/problems/minimum-rounds-to-complete-all-tasks/description/)说明题解思路hash表 Codehash表 说明 给你一个下标从 0 开始的整数数组 tasks &#xff0c;其中 tasks[i] …

6 逻辑回归评分卡

6 逻辑回归评分卡 学习目标 掌握KS值的计算方法知道评分映射方法1 模型构建流程 1.1 实验设计 新的模型能上线一定要比原有方案有提升,需要通过实验证明 冷启动业务初期成长期波动期策略调整新增数据源人工审核人工审核新旧模型对比新旧模型对比避免迭代模型新旧模型对比规…

数据结构【顺序表】

文章目录 1.顺序表的概念线性表物理结构逻辑结构 2.顺序表的分类2.1静态顺序表2.2动态顺序表 3.顺序表接口的实现头文件(SQList.h)如下源文件初始化顺序表销毁顺序表插入扩容尾插头插 封装扩容函数删除尾删头删 查找元素在指定位置前插入数据情况一(指定的位置不是首元素)情况二…

[笔记]docker入门《四》之 dockerfile和docker-compose

文章目录 前言dockerfiledocker-compose总结 前言 dockerfile dockerfile主要是构建镜像. dockerfile的作用是从无到有的构建镜像。它包含安装运行所需的环境、程序代码等。这个创建过程就是使用 dockerfile 来完成的。 dockerfile的作用是从无到有的构建镜像。它包含安装运行…

JAVA面试库

1、基础 1.1、面向对象编程有哪些特性 1、抽象 抽象就是对同一个目标的共有的属性、特征、方法、功能、行为等进行抽取并归纳总结&#xff0c;它是一种将复杂现实简单化为模型的过程&#xff0c;它关注的是对象行为&#xff0c;而不用关注具体的实现细节。 在面向对象编程中…

中关村论坛 | 区块链与隐私计算论坛倒计时1天!

「区块链与隐私计算论坛」 倒计时1天&#xff01; 地址&#xff1a;中关村国家自主创新示范区会议中心&#xff08;新建宫门路2号&#xff09;万春厅 时间&#xff1a;2024年4月27日&#xff0c;下午14:30-17:00 本次论坛围绕释放数据要素价值深入探讨如何将区块链与隐私计算…

【Docker学习】查询容器镜像的docker search

这个命令是使用Docker的必备技能。我们使用的各种官方镜像&#xff0c;一般都能通过这个命令找到。 命令&#xff1a; docker search 描述&#xff1a; 在Docker Hub上查找镜像。Docker Hub是为开发者和开源贡献者设计的容器镜像注册中心&#xff0c;它允许用户查找、使用和…

用python写了一个把iptv节目列表换成txt 适配diyp

这几天抓取了很多iptv&#xff0c;需要列表&#xff0c;列表太多麻烦&#xff0c;所以编写个小程序python转换一下 代码如下 # 读取原始文本文件 with open(iptv.txt, r, encodingutf-8) as file:lines file.readlines()# 定义存储频道信息的字典 channels {"央视频道&…

MySQL中order by排序时,数据存在null,排序在最前面

order by排序是最常用的功能&#xff0c;但是排序有时会遇到数据为空null的情况&#xff0c;这样排序就会乱了&#xff0c;这里以MySQL为例&#xff0c;记录我遇到的问题和解决思路。 sql 排序为 null 值问题&#xff1a; 排序时我们用 receive_date(一个统计的时间&#xff…

【js刷题:数据结构链表之设计链表】

设计链表 一、题目二、题解 一、题目 二、题解 // 定义节点类&#xff0c;每个节点都有一个值和一个指向下一个节点的引用 class LinkNode{constructor(val,next){ // 构造函数&#xff0c;接收节点值和下一个节点的引用this.valval // 节点的值this.nextnext // 指…

欣赏一个尚未关闭的python运行时bug

这是一个语言的运行时错误&#xff0c;在linux环境&#xff0c;跨语言使用共享内存时&#xff0c;会触发。它会在python程序退出时&#xff0c;自行销毁sharedMemory&#xff0c;即便此时还有其他的进程在使用——这会让C/Python跨进程调用几乎没有办法进行。 python程序运行完…

校园科普气象站的工作原理

TH-XQ3校园科普气象站是学校为了进行气象科普教育而设立的一种特殊设施。它不仅是一个能够实时监测和记录各种气象参数的气象站&#xff0c;更是一个促进学生对气象科学兴趣和理解的重要平台。 校园科普气象站通常包括一系列的气象观测设备和相关的科普设施。这些设备包括但不限…

Ubuntu20.04调试功能包的一些报错解决办法【更新中2024.05.14】

一、Could not find a package configuration file provided by “catkin_virtualenv” 解决办法&#xff1a; sudo apt install ros-noetic-catkin-virtualenv二、 ERROR: Could not find a version that satisfies the requirement pip-tools5.1.2 (from versions: none) …

electron 使用两个页面(额外添加一个html文件)

需求&#xff1a;打开窗口 (加载本地的html页面) 并播放视频资源 环境 electron 28.1.3 electron-forge 7.2.0 思路&#xff1a;因为要加载新弹出一个窗口并播放资源&#xff0c;可以自己加载一个外部的页面或者加载一个本地的页面&#xff0c;使用本地的会好些。让electron-…

【制作100个unity游戏之26】unity2d横版卷轴动作类游戏5(附带项目源码)

最终效果 系列导航 文章目录 最终效果系列导航前言三段攻击攻击设置只对敌人造成伤害限制可以移动攻击问题 角色连续按四下攻击&#xff0c;最后会多a一下问题&#xff1a;站在原地连续攻击野猪&#xff0c;只有第一下攻击野猪才掉血&#xff0c;后面的攻击野猪不掉血源码完结 …

深⼊理解指针(5)

目录 1. 回调函数是什么&#xff1f;1.1 使用回调函数修改 2. qsort使⽤举例2.1 使⽤qsort函数排序整型数2.2 使⽤qsort排序结构数据按年龄排序2.3 使⽤qsort排序结构数据按名字排序2.4整体代码 3. qsort函数的模拟实现3.1 整型数组的实现3.2 结构体按名字排序实现3.3 结构体按…

蓝桥杯练习系统(算法训练)ALGO-941 P0601字符删除

资源限制 内存限制&#xff1a;256.0MB C/C时间限制&#xff1a;1.0s Java时间限制&#xff1a;3.0s Python时间限制&#xff1a;5.0s 编写一个程序&#xff0c;先输入一个字符串str&#xff08;长度不超过20&#xff09;&#xff0c;再输入单独的一个字符ch&#xff0c…

java数据结构与算法(二叉树中序遍历)

前言 二叉树的中序遍历是一种特定的遍历方法&#xff0c;按照左子树、根节点、右子树的顺序进行遍历。如果二叉树为空&#xff0c;则遍历结束并返回&#xff1b;否则&#xff0c;首先递归遍历左子树&#xff0c;然后访问根节点&#xff0c;最后递归遍历右子树 实现原理 中序…

JavaScript事件监听

JavaScript事件监听是指在某个元素上监听特定事件的触发&#xff0c;并在事件触发时执行相应的函数。事件可以是用户的鼠标操作、键盘操作、页面加载等等。 在JavaScript中&#xff0c;可以使用addEventListener方法来添加事件监听器。它接受三个参数&#xff1a;事件类型、要…

第3章Spring Boot进阶,开发社区核心功能【仿牛客网社区论坛项目】

第3章Spring Boot进阶&#xff0c;开发社区核心功能【仿牛客网社区论坛项目】 前言推荐项目总结第3章Spring Boot进阶&#xff0c;开发社区核心功能1.过滤敏感词2.发布帖子3.帖子详情4.事务管理5.显示评论6.添加评论7.私信列表8.发送私信9.统一处理异常10.统一记录日志 最后 前…