【Java】CAS详解

一.什么是CAS

CAS(compare and swap) 比较并且交换. CAS是一个cpu指令,是原子的不可再分.因此基于CAS就可以给我们编写多线程的代码提供了新的思路---->使用CAS就不用使用加锁,就不会牵扯到阻塞,也称为无锁化编程
下面是一个CAS的伪代码:
在这里插入图片描述
address是一个内存地址,expectValue和swapValue都表示一个寄存器中的值,比较比较的是内存地址中的值和expectValue的值是否一样,交换就是:如果一样就把address和swapValue的值进行交换.如果不相同,则一直循环对比,直到成功为止.
通俗来讲就是:比较当前工作内存中的值和主内存中的值,如果相同则执行规定操作,否则继续比较直到主内存和工作内存中的值一致为止.
举一个详细的例子:(count++)
现在有两个线程t1和t2并发执行count++这个操作,count++需要牵扯到三个操作,①读取:从内存中读取当前计数器的值.②.读取到的值+1.③.写入:将+1的值写回到内存的计数器中.

  • 开始执行,address中的值为0,t1线程执行完①操作,此时读取到的值是0(expectValue的值为0),address中的值也为0.此时能够进行++操作.
  • 但是系统调度t2线程执行执行①②③.也可以执行count++ 的操作.执行完后内存地址中的值为1.
  • 然后线程调度回t1,此时发现address(1)和expect Value(0)的值不一样,此时就不能进行++操作.
  • 重新读取内存中当前计数器的值,读取过后expectValue的值变为了1,此时线程1就能进行++操作了.
    此时就能保证线程安全了.
下面我将在Java中用CAS来解决count++这个问题

代码实例:

import java.util.concurrent.atomic.AtomicInteger;

//使用原子类解决问题
public class Test11 {
    private static AtomicInteger count = new AtomicInteger(0);
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(()->{
            for(int i = 0;i<5000;i++){
                count.getAndIncrement();
                //++count
                //count.incrementAndGet();
            }
        });
        Thread t2 = new Thread(()->{
            for(int i = 0;i<5000;i++){
                count.getAndIncrement();
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("count = " +count.get());
    }
}

运行结果:
在这里插入图片描述

CAS的优缺点

  • 优点:
  1. 避免竞态条件:CAS操作通过比较内存位置的当前值与预期值,只有在它们相等时才更新该内存位置的值,从而避免了多线程下的竞态条件。
  2. 高并发性能:由于CAS操作不会引起线程阻塞和上下文切换的开销,因此它具有很高的并发性能,适用于高并发场景。
  3. 适应性强:CAS操作可以应用于各种高并发场景中,是Java并发包(java.util.concurrent)的基础。
  • 缺点:
  1. ABA问题:在执行CAS操作时,如果共享变量的值从A变为B又变回了A,CAS操作无法区分这种情况,因为它只比较值而不是版本或标识符。
  2. 硬件限制:CAS操作的性能可能受到硬件指令集的限制,不同的硬件平台可能对CAS的支持程度不同。
  3. 高CPU消耗:在高并发环境下,CAS操作可能会因为多次尝试而消耗较高的CPU资源。
  4. 不支持多步操作:CAS操作只能保证单个共享变量的原子性更新,对于需要多步操作的复合动作,需要通过循环CAS或其他机制来实现。

CAS的ABA问题.

什么是ABA问题:

如果一个值原来是A,变成了B,又变成了A,那么使用CAS进行检查时会发现它的值没有发生变化,但是实际上却变化了.
举个例子:假设去银行取钱,初始情况下账余额为1000 , 现在要取500!!取钱的时候ATM卡了,我按了一下卡了,就又摁了一下,此时就产生了两个线程来进行扣款,假设是由CAS模式来进行扣款=====>此时如果出现第三个线程在t1线程执行完CAS的第一步,t2线程执行完扣款操作了,此时第二次扣款操作也会执行:此时就会产生bug.相当于两次取款都变得有效了.

对于ABA问题的解决方法:

①:约定数据变化是单向的(只能增加或只能减少),不能是双向的(又能增加又能减少)
②.对于本身就需要双向变化的数据,可以给它引入一个版本号,版本号的数字是只能增加不能减少的

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

电脑便签软件怎么用 纯分享几款电脑便签软件

在当今快节奏的生活中&#xff0c;随时记录灵感、待办事项或重要信息变得越来越重要。电脑便签软件成为了我们日常生活中不可或缺的工具之一。本文将介绍几款常见的电脑便签软件&#xff0c;并分享它们的使用方法&#xff0c;帮助读者更好地管理信息和提高工作效率。 1. 敬业签…

【ESP32S3 Sense接入语音识别+MiniMax模型+TTS模块语音播报】

【ESP32S3 Sense接入语音识别MiniMax模型TTS模块语音播报】 1. 前言2. 功能模块概述2.1 语音接入2.2 大模型接入2.3 TTS模块接入 3. 先决条件3.1 环境配置3.2 所需零件3.3 硬件连接步骤 4. 核心代码4.1 源码分享4.2 代码解析 5. 上传验证5.1 对话测试5.2 报错 6. 总结 1. 前言 …

NumPy的ndarray常用属性和索引你学会了吗

1.ndarray的4个重要属性 ndim&#xff1a;返回数组的维度数。例如&#xff0c;一维数组的ndim为1&#xff0c;二维数组的ndim为2 shape&#xff1a;返回数组的形状&#xff0c;即各个维度的大小。例如&#xff0c;对于一个二维数组&#xff0c;shape会返回一个包含行数和列数的…

Docker in Docker原理与实战探索

博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a; 《面试题大全》 — 面试准备的宝典&#xff01;《IDEA开发秘籍》 — 提升你的IDEA技能&#xff01;《100天精通鸿蒙》 …

使用CyberRT写第一个代码

0. 简介 计算框架是自动驾驶系统中的重中之重&#xff0c;也是整个系统得以高效稳定运行的基础。为了实时地完成感知、决策和执行&#xff0c;系统需要一系列的模块相互紧密配合&#xff0c;高效地执行任务流。由于各种原因&#xff0c;这些模块可能位于不同进程&#xff0c;也…

Android Studio 打开Logcat界面

在平时调试过程中查看调试日志需要打开 Android Studio Logcat界面。 每次安装AS都会忘记&#xff0c;自己备注一下。 AS->View->Tool Windows->Logcat

Windows12安装Docker

环境及工具&#xff08;文末提供&#xff09; Docker Desktop Installer.exe &#xff08;官网&#xff09; 一、查看windows相关配置 查看是否开启相应的功能&#xff0c;如果没有需要开启&#xff0c;然后重启电脑 打开任务管理器&#xff08;CTRLSHIFTESC&#xff09;-&g…

【Linux】进程控制详解

目录 前言 进程创建 认识fork 写时拷贝 再谈fork 进程终止 进程退出码 用代码来终止进程 常见的进程终止的方式 exit _exit 进程等待 进程等待的必要性 进程等待的方式 wait waitpid 详解status参数 详解option参数 前言 本文适合有一点基础的人看的&#…

【JavaScript】函数 ⑥ ( 使用 arguments 获取所有实参 | arguments 内置对象 | 伪数组概念 )

文章目录 一、使用 arguments 获取所有实参1、arguments 内置对象2、伪数组概念3、arguments 实参遍历4、arguments 代码示例 - 基本使用5、arguments 代码示例 - 遍历实参 一、使用 arguments 获取所有实参 1、arguments 内置对象 在 定义 JavaScript 函数 时 , 有时 不确定 形…

Spring定义Bean对象笔记(二)

前言&#xff1a;上一篇记录了通过XML文件来定义Bean对象&#xff0c;这一篇将记录通过注解和配置类的方式来定义Bean对象。 核心注解&#xff1a; 定义对象&#xff1a;Component,Service,Repository,Controller 依赖注入&#xff1a; 按类型&#xff1a;Autowired 按名称&am…

cesium 加载mapbox底图 黑色主题底图 84底图

cesium提供MapboxStyleImageryProvider&#xff0c;加载mapbox的影像图层&#xff0c;底图是84坐标系。 viewer.imageryLayers.addImageryProvider(new Cesium.MapboxStyleImageryProvider({styleId: dark-v11,accessToken: mapbox的token})); 效果图&#xff1a;加载mapbox黑…

简单使用bootstrap-datepicker日期插件

目录 下载datepicker 方式一&#xff1a; 方式二&#xff1a; 下载依赖 下载bootstarp.js 下载jquery 使用示例 日期选择 单独选择年 单独选择月 单独选择日 设置截止日期 设置默认日期 总结 下载datepicker 方式一&#xff1a; 下载地址 GitHub - uxsolution…

java运行时内存

从jdk1.7以及以后&#xff0c;静态变量和常量池存在堆空间。

关于搭建电商独立站跨境电商接入主流电商平台API商品接口对于商品功能模块的巨大应用

功能设计 首先我们来看下mall项目中商品功能的设计&#xff0c;主要包括商品管理、添加\编辑商品、商品分类、商品类型、品牌管理等功能&#xff0c;这里的功能同时涉及前台商城和后台管理系统。 商品管理【接入主流电商平台商品API接口丰富自建商城商品】 在mall项目的后台管…

JavaScript基础(5)之对象的方法和调用

JavaScript基础5之对象的方法和调用 对象对象使用语法属性和访问方法和调用null遍历对象 内置对象Math属性方法 基本数据类型和引用数据类型堆栈空间分配区别&#xff1a;简单类型的内存分配复杂类型的内存分配 对象 对象是 JavaScript 数据类型的一种&#xff0c;之前已经学习…

填谷式无源PFC电路

目录&#xff1a; 1、概述 2、原理 1、概述 如果不采用PFC&#xff0c;那么典型开关模式电源的功率因数约为0.6&#xff0c;因而会有相当大的奇次谐波失真(第三谐波有时和基本谐波一样大)。令功率因数小于1以及来自峰值负载的谐波减少了运行设备可用的实际功率。为运行这些低…

书生·浦语2.0体系技术报告

前言 本文是书生浦语二期实战营课程视频笔记&#xff0c;如果需要详细视频教程可自行搜索。 InternLM2 InternLM2-Base 高质量和具有很强可塑性的模型基座&#xff0c;是模型进行深度领域适配的高质量起点InternLM2 在Base基础上&#xff0c;在多个能力方向进行了强化&#x…

WSL安装与使用

开启之后&#xff0c;会提示你重启电脑才能使配置生效&#xff0c;我们重启即可。 电脑重启后&#xff0c;打开Microsoft Store搜索WSL&#xff0c;既可以看到支持的操作系统&#xff0c;我们选择Ubuntu即可&#xff0c;我们选择第一个就可以。 随后我们打开&#xff0c;发现报…

提升办公效率,一起了解流程自定义表单优势

提高办公效率&#xff0c;可以一起了解低代码技术平台。对于很多中小型企业而言&#xff0c;低代码技术平台及流程自定义表单优势突出&#xff0c;是助力企业实现流程化办公&#xff0c;实现数字化转型的得力助手。流辰信息是专业研发开发平台、数据治理、数据分析等产品的服务…

软考 系统架构设计师系列知识点之云原生架构设计理论与实践(13)

接前一篇文章&#xff1a;软考 系统架构设计师系列知识点之云原生架构设计理论与实践&#xff08;12&#xff09; 所属章节&#xff1a; 第14章. 云原生架构设计理论与实践 第3节 云原生架构相关技术 14.3.2 云原生微服务 1. 微服务发展背景 过去开发一个后端应用最为直接的方…