【JavaEE初阶系列】——CAS

news2025/1/11 11:40:49

目录

🎈什么是 CAS

📝CAS 伪代码 

🎈CAS 是怎么实现的

🎈CAS 有哪些应用

🚩实现原子类

🌈伪代码实现:

🚩实现自旋锁

🌈自旋锁伪代码

🎈CAS 的 ABA 问题

🚩什么是 ABA 问题

🚩ABA 问题引来的 BUG

🎈CAS相关面试题


🎈什么是 CAS

CAS: 全称 Compare and swap ,字面意思 :” 比较并交换 ,一个 CAS 涉及到以下操作:
我们假设内存中的原数据V,旧的预期值A,需要修改的新值B。
  • 1. 比较 A 与 V 是否相等。(比较)
  • 2. 如果比较相等,将 B 写入 V。(交换)
  • 3. 返回操作是否成功。

CAS 比较交换的是 内存 和 寄存器。比如有一个内存M,现在有俩个寄存器A,B.

CAS(M,A,B)如果M和A的值相同的话,就把M和B的值交换,同时整个操作返回true.

                     如果M和A的值不同的话,无事发生,同时整个操作返回false.


📝CAS 伪代码 

伪代码 ,代码是不能真正编译执行(不符合语法要求) 认识到逻辑是啥样的。

下面写的代码不是原子的, 真实的 CAS 是一个原子的硬件指令完成的. 这个伪代码只是辅助理解
CAS 的工作流程.
boolean CAS(address, expectValue, swapValue) {
if (&address == expectedValue) {
  &address = swapValue;
       return true;
  }
   return false;
}
两种典型的不是 "原子性" 的代码
  • 1. check and set (if 判定然后设定值) [上面的 CAS 伪代码就是这种形式]
  • 2. read and update (i++) [之前我们讲线程安全的代码例子是这种形式]

当多个线程同时对某个资源进行 CAS 操作,只能有一个线程操作成功,但是并不会阻塞其他线程 , 其他线程只会收到操作失败的信号。
CAS 可以视为是一种乐观锁. (或者可以理解成 CAS 是乐观锁的一种实现方式)

CAS其实是一个cpu指令(一个cpu指令,就能完成上述比较交换的逻辑)单个cpu指令,是原子的!就可以使用CAS完成一些操作,进一步的替代"加锁"。——这样就给编写线程安全的代码,引入了新的思路。

基于CAS实现线程安全的方式,也称为"无锁编程",

  • 优点:保证线程安全,同时避免阻塞(效率)
  • 缺点:代码会更复杂,不好理解,只能够适合一些特定的场景,不如加锁方式更普实。

🎈CAS 是怎么实现的

CAS本质上是cpu提供的指令——》又被操作系统封装,提供成api,然后又被JVM,也提供成api——》程序员就可以使用了。

针对不同的操作系统, JVM 用到了不同的 CAS 实现原理,简单来讲:
  • java CAS 利用的的是 unsafe 这个类提供的 CAS 操作;
  • unsafe CAS 依赖了的是 jvm 针对不同的操作系统实现的 Atomic::cmpxchg
  • Atomic::cmpxchg 的实现使用了汇编的 CAS 操作,并使用 cpu 硬件提供的 lock 机制保证其原子性。
简而言之,是因为 硬件予以了支持,软件层面才能做到。

等到后面会更加能理解。


🎈CAS 有哪些应用

🚩实现原子类

int 进行++,不是原子的(load,add,save)三步骤。

AtomicInteger,基于CAS的方式对int进行封装,此时进行++,就是原子的了。(++操作是基于CAS指令实现的)

标准库中提供了 java.util.concurrent.atomic 包, 里面的类都是基于这种方式来实现的.
典型的就是 AtomicInteger 类. 其中的 getAndIncrement 相当于 i++ 操作.
在java中,有些操作是偏底层的操作,偏底层的操作在使用的时候有更多的注意事项,稍有不慎就容易写出问题。这些操作,就会放到unsafe中进行归类。
unsafe代表有更多的注意事项,稍有不慎就写错。(就比如在导航的时候,遇到事故多发地方就会提醒警告信息)
我们看到原子类内部没有使用synchronized加锁使用。
native是本地方法,compareAndSwapInt比较和交换,JVM源码中,使用c++实现逻辑,底层的操作。
从上面流程我们可以看到,CAS中cpu指令 ,先是通过系统进行封装,提供了api(getAndSetInt),然后JVM进行封装提供api(compareAndSwapInt)。而原子类是基于CAS实现的。
  • 原子类里面基于CAS实现的。                                                                                          通过利用指令原子性逻辑获取锁实现原子性操作。

🌈伪代码实现:

class AtomicInteger {
    private int value;
    public int getAndIncrement() {
        int oldValue = value;
        while ( CAS(value, oldValue, oldValue+1) != true) {
            oldValue = value;
       }
        return oldValue;
   }
}

初始情况下,value的值是0,俩次自增结果是2.

如果俩者相等,就返回true,并且让oldValue+1赋值给value,让value+1,如果不相等就得让value赋值给oldvalue,然后进行++操作。

所以我们之前所说的”线程不安全“本质上是进行自增的过程中,穿插执行了。

CAS也是让这里的自增,不要穿插执行,核心思路和加锁是类似的,加锁是通过阻塞的方式,避免穿插,CAS则是会通过重试的方式,避免穿插。


🚩实现自旋锁

基于 CAS 实现更灵活的锁 , 获取到更多的控制权 .

🌈自旋锁伪代码

public class SpinLock {
    private Thread owner = null;
    public void lock(){
        // 通过 CAS 看当前锁是否被某个线程持有. 
        // 如果这个锁已经被别的线程持有, 那么就自旋等待. 
        // 如果这个锁没有被别的线程持有, 那么就把 owner 设为当前尝试加锁的线程. 
        while(!CAS(this.owner, null, Thread.currentThread())){
       }
   }
    public void unlock (){
        this.owner = null;
   }
}

记录当前这个锁被哪个线程获取到了,如果是null,表示未加锁状态。


🎈CAS 的 ABA 问题

🚩什么是 ABA 问题

ABA 的关键问题 : 是通过值"没有发生改变"来作为”没有其他线程穿插执行“判定依据。
但是这种判定方式不够严谨,更极端的情况下,可能有另一个线程穿插进来,把值从A->B->A,针对第一个线程来说,看起来好像是这个值,没变,实际上已经被穿插执行了。
比如,买个手机,买到的是一个”二手的,翻新的设备“。翻新机,也不是不能用,里面可能会有一些暗伤。
ABA问题如果真的出现了,其实大部分情况下是不会产生bug的,就相当于买到二手设备,也是能用的,虽然另一个线程穿插执行了,由于值又改回来了,此时逻辑上也不一定会产生bug。

🚩ABA 问题引来的 BUG

假设这个场景,我去ATM取钱,我本身的账户1000,我想要取500,我再取钱的过程中,出现bug了,我按下取钱按钮的时候,没反应,又按了一下,此时就产生了俩个线程进行扣款操作。

由于t3线程正好又在这个节骨眼上转来了500,与时我的余额又是1000了,就会导致t1线程也能扣款。此时我预期取500,实际上扣了1000.

大部分情况下ABA问题其实没啥大事,但是有一些极端情况,会使ABA出现bug,只要让判定的数值,按照一个方向增长即可。有增有减,就可能出现ABA,只是增加,或者只是减少,针对像账户余额这样概念,本身就应该要能增有减,可以引入一个额外的变量,版本号,约定每次修改余额就让版本号自增,此时在使用CAS判定的时候,就不是直接判定余额了,而是判定版本号,看版本号是否是变化了,如果版本号不变,注定没有线程穿插了执行。


🎈相关面试题

1) 讲解下你自己理解的 CAS 机制
compareAndSwap 比较并且交换,相当于一个原子操作,同时完成 , "读取内存, 比
较是否相等, 修改内存" 这三个步骤. 本质上需要 CPU 指令的支撑。通过利用指令的原子性从而避免获取锁实现了原子性操作。
2) ABA问题怎么解决?
给要修改的变量添加一个版本号,在 CAS 比较数据当前值和旧值的同时, 也要比较版本号是否符合预期.如果发现当前版本号和之前读到的版本号一致, 就真正执行修改操作, 并让版本号自增; 如果发现当前版本号比之前读到的版本号大, 就认为操作失败

难道父母眼里只有学习学习嘛??

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

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

相关文章

黄金票据制作-新手向

黄金票据制作 文章目录 黄金票据制作0x01 前言0x02 黄金票据的制作一、靶场搭建二、收集制作信息获取域名称获取域SID值获取域用户krbtgt密码hash值 二、制作票据 0x03 验证票据有效性 0x01 前言 最近,我学习了内网渗透的相关知识,其中包括了黄金票据的…

浏览器页面缓存机制

HTTP缓存机制的核心思想是,对于已经请求过的资源,如果其在服务器上没有发生变化,那么浏览器就可以直接从本地缓存中获取这些资源,而无需再次向服务器发送请求。 强缓存 就是确定可用的缓存 浏览器和和服务器对每个缓存资源先商量一…

fread和fwirte函数

✨✨ 欢迎大家来到莉莉的博文✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 一、fread函数 ——>从文件流中读取二进制数据到ptr指向的数组 从流(二进制文件)中读取数据块 ptr:指向大小至…

基于JSP在线订花系统

基于JSP在线订花系统的设计与实现 摘要 近年来,随着人们对于生活品质的重视度日益提升,鲜花的需求量也在不断增加同时带动了鲜花电商的飞速发展和应用,平台化的鲜花交易模式也逐渐从传统的鲜花销售转型为个性化鲜花定制。同时随着鲜花速递行…

基于 RisingWave 和 ScyllaDB 构建事件驱动应用

概览 在构建事件驱动应用时,人们面临着两大挑战:1)低延迟处理大量数据;2)实现流数据的实时摄取和转换。 结合 RisingWave 的流处理功能和 ScyllaDB 的高性能 NoSQL 数据库,可为构建事件驱动应用和数据管道…

MTMT:构建比特币生态平行世界 打造铭文生态繁荣

近年来,随着铭文市场的火爆以及比特币ETF成功通过,比特币生态正经历着一场复兴,尤其是铭文市场作为新一代Web3的叙事,带来了全新的生产方式,可以预见,铭文就像流动性挖矿对于上一轮DeFi Summer的推动一样会…

KNN算法 | K近邻:KD Tree、球树、KNN数据下采样策略

目录 一. KNN算法实现方式1. 蛮力实现(brute)2. KD树(kd_tree)3. 球树(ball_tree) 二. KD Tree算法1. 构建方式2. KD Tree 查找最近邻 三. 球树(Ball Tree)1. 构建方式 四. KNN评价1. 优点2. 缺点 五. 延申1. KNN数据下采样策略策略1策略2策略3策略4 Condensed Nearest Neighbo…

Macs Fan Control Pro--精准掌控Mac风扇,优化散热新选择

Macs Fan Control Pro是一款专为Mac电脑设计的高级风扇控制工具。它具备强大的温度监测能力,可以实时监测Mac电脑各个核心组件的温度,并通过直观的界面展示给用户。同时,用户可以根据个人需求自定义风扇速度,或者选择预设的自动风…

蓝桥杯算法题-图形排版

题目描述 小明需要在一篇文档中加入 N 张图片,其中第 i 张图片的宽度是 Wi,高度是 Hi。   假设纸张的宽度是 M,小明使用的文档编辑工具会用以下方式对图片进行自动排版: 1. 该工具会按照图片顺序,在宽度 M 以内&…

LabVIEW转动设备故障诊断系统

LabVIEW转动设备故障诊断系统 随着工业自动化技术的不断进步,转动设备在电力、化工、船舶等多个行业中扮演着越来越重要的角色。然而,这些设备在长期运行过程中难免会出现故障,如果不能及时诊断和处理,将会导致生产效率下降&…

【JavaSE】一维数组和二维数组详解

欢迎关注个人主页:逸狼 创造不易,可以点点赞吗~ 如有错误,欢迎指出~ 目录 一维数组 基本语法 初始化 遍历和打印 数组是引用型变量 基本类型变量与引用类型变量的区别 null 数组传参和返回 总结 二维数组 基本语法 初始化 遍历和打印 一维数组…

排序——非基于比较的排序

本专栏和大家分享关于排序的算法,其中有插入排(直接插入排序和希尔排序)、选择排序(直接选择排序和堆排)、交换排序(冒泡排序和快速排序)、归并排序以及其他非基于比较的排序 本文与大家分享非基于比较的排…

透视未来安全:PIR技术引领数据隐私新时代

1.隐语实现PIR总体介绍 隐语实现的Private Information Retrieval (PIR) 是一种隐私增强技术,它使用户能够在不暴露他们实际查询内容的情况下从远程服务器数据库中检索所需信息。以下是隐语在实现PIR方面的概要说明和技术特点: 基本概念: PI…

移动硬盘无法读取打不开?原因分析与解决方案

一、移动硬盘遭遇困境:无法读取打不开 在数字信息爆炸的时代,移动硬盘成为了我们储存和传输数据的重要工具。然而,当移动硬盘突然无法读取打不开时,我们往往会感到手足无措。插入电脑后,移动硬盘的指示灯可能正常闪烁…

算法系列--动态规划--特殊的状态表示--分析重复子问题

💕"轻舟已过万重山!"💕 作者:Lvzi 文章主要内容:算法系列–算法系列–动态规划–特殊的状态表示–分析重复子问题 大家好,今天为大家带来的是算法系列--动态规划--特殊的状态表示--分析重复子问题 一.组合总数IV 链接…

dubbo下

dubbo 集成springboot 配置文件 controller 启动类 注册中心宕机 负载均衡 zookeeper注册中心 dubbo原理 dubbo架构 各层说明 增强spi原理

Native Instruments Kontakt 7 for Mac v7.9.0 专业音频采样

Native Instruments Kontakt 7是一款强大的软件采样器,它允许用户从各种来源采样音频并进行编辑和处理。它包含大量预设采样库,包括乐器、合成器、鼓组和声音效果等。此外,Kontakt 7还允许用户创建自己的采样库,以便根据自己的需要…

2-Prometheus监控主机

1 介绍 Prometheus 使用 node_exporter 服务程序监控 Linux 主机。 2 部署 2.1 下载 官方下载地址 https://prometheus.io/download/ 找到 node-export 下载即可 curl -o node-exporter.tar.gz -L https://github.com/prometheus/node_exporter/releases/download/v1.7.0/…

Linux非root用户安装mysql5.7

1、下载安装包MySQL :: Download MySQL Community Server 点击Archives 我下载的是5.7.27版本,linux主机直接选择linux-Generic即可,选择第一个包下载即可 2、安装mysql 解压 shell> tar xzvf mysql-5.7.31-linux-glibc2.12-x86_64.tar.gz shell&g…

idea中Git项目遇到“Filename too long”错误 与 配置Git的ssh证书

一:“Filename too long”问题解决办法 错误信息: fatal: cannot create directory at xxxx: Filename too long warning: Clone succeeded, but checkout failed. You can inspect what was checked out with git status and retry with git restore …