JAVA并发编程JUC包之CAS原理

news2025/1/11 18:46:47

在JDK 1.5之后,java api中提供了java.util.concurrent包,简称JUC包。这个包定义了很多我们非常熟悉的工具类,比如原子类AtomicXX,线程池executors、信号量semaphore、阻塞队列、同步器等。日常并发编程要用的熟面孔基本都在这里。

       首先,Atomic包,原子操作类,提供了用法简单、性能高效、最重要是线程安全的更新一个变量。支持整型、长整型、布尔、double、数组、以及对象的属性原子修改,支持种类非常丰富。

        之前的文章《JAVA并发编程volatile核心原理》说过,volatile只是解决了多线程的可见性和有序性问题,原子性问题并没有解决。 在这里volatile+Atomic原子类可以完美实现多线程共享变量的安全修改。

1、Atomic原子类并发修改共享变量demo

package lading.java.mutithread;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * 多线程并发下,使用原子类+volatile进行操作,实现共享变量修改并发安全
 * 10个线程并发对共享变量count进行求和,每个线程求和100次,最后结果是1000
 */
public class Demo005AtomicCount {
    public static volatile AtomicInteger count = new AtomicInteger(0);

    public static void main(String[] args) {
        int threadNum = 10;
        ExecutorService threadPool = Executors.newFixedThreadPool(threadNum);
        for (int i = 0; i < threadNum; i++) {
            //10个线程并发对共享变量count进行求和,每个线程求和100次
            threadPool.submit(() -> {
                System.out.println(Thread.currentThread().getName() + "开始对count求和100次");
                for (int j = 0; j < 100; j++) {
                    count.getAndIncrement();
                }
            });
        }
        threadPool.shutdown();
        System.out.println("最后count的值:" + count.get());
    }
}

结果,刚好就是1000.

2、说一下CAS原理

      CAS 英文原名compare AND swap(比较再置换)。在JUC包中,大量应用了CAS的原理实现,比如AQS ,concurrentHashMap都有应用CAS技术。可以说,synchronized、volatile、CAS就是JAVA并发编程里的基石。

       CAS本质是一条CPU指令,fun(address,oldValue,newValue),其中Address就是本次操作要读取旧值以及修改为新值的内存地址,oldValue就是该内存地址之前的旧值,newValue就是本次操心希望写入的新值。

核心原理:判断内存地址address当前的值是否与oldValue相同,如果相同就更新该内存地址的值为newValue。

先看一下源码,这个AtomicInteger +1源码:

public final int getAndAddInt(Object var1, long var2, int var4) {
        int var5;
        do {
            var5 = this.getIntVolatile(var1, var2);
        } while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));

        return var5;
    }

然后compareAndSwapInt的源码如下:
public final native boolean compareAndSwapInt(Object var1, long var2, int var4, int var5);

       这里this.compareAndSwapInt(var1, var2, var5, var5 + var4));可以看到其实是4个变量,实际就是我们常说的三个变量,因为Object var1表示这个对象内存地址,var2只是偏移量。

       这个偏移量有什么用呢?继续看unsafe c++的源码,发现,其实和我们之前讲的JVM内存模型是一样的,一个对象有自己的地址,对象在内存中会占据一段内存区域,这一片区域会有对象的各种信息,比如锁状态,对象头信息,最重要的是还有我们这个对象的值。那如何找到这个值的内存地址呢?

      这时候就是需要这个long var2 偏移量。通过这个var2才可以读到当前变量的旧值old。然后拿这个old和预期的oldValue去比对,相等了就把newValue更新到值的内存地址。

3、CAS有什么缺陷吗

        在2的核心原理源码看到,很明显,如果发现从内存地址里值不是预期的oldValue,那就陷入了死循环。CAS就是一个while循环判断内存地址值是否等于预期值,不等于就继续循环。

3.1 CAS一直不成功,就会导致CPU自旋时间过长,CPU开销过大。

3.2 A->B->A问题,俗称ABA问题。

        举个例子,有个变量name="A",有个线程想把name从A修改为C。但是很不幸在次过程中被别的线程修改为B后,又再修改为A,它才能执行。再具体一点就是:有个人存款有100块,想去取出来,但是取款机显示余额是0,无法取现;过了一会,发现余额是100,这才能取现。这个诡异的过程就是ABA的问题,CAS对变量预期旧值的循环变化是无感的。

4、如何解决CAS的ABA问题

我们先看一下ABA的demo。

package lading.java.mutithread;

import java.util.concurrent.atomic.AtomicInteger;

/**
 * CAS ABA问题复现
 * count 原值100
 */
public class Demo006CasBug {
    public static volatile AtomicInteger count = new AtomicInteger(100);

    public static void main(String[] args) throws InterruptedException {
        //线程1先修改为200,然后马上修改为100
        Thread thread1 = new Thread(() -> {
            count.compareAndSet(100, 200);
            count.compareAndSet(200, 100);
            System.out.println(Thread.currentThread().getName() + "完成对count的从100修改为200,然后再修改为100");
        });
        Thread thread2 = new Thread(() -> {
            count.compareAndSet(100, 300);
            System.out.println(Thread.currentThread().getName() + "对count的从100修改为" + count.get());
        });
        thread1.start();
        thread2.start();
        thread1.join();
        thread2.join();
    }
}

结果是线程1无感的修改为300,实际上期间线程0已经修改了2次。

JDK提供了AtomicStampedReference来解决。解决demo:

package lading.java.mutithread;
import java.util.concurrent.atomic.AtomicStampedReference;

/**
 * 解决CAS ABA问题
 * 用AtomicStampedReference,给原子类增加一个时间戳6666,这样每次修改需要增加判断预期的时间戳
 * 类似数据库mysql更新一个记录,update set col1='' where col2=oldVal and timestamp=oldVal
 * 在cas基础上增加一个数据时间戳,确保数据未更新修改过。
 */
public class Demo007CasBugFix {
    public static AtomicStampedReference<Integer> count = new AtomicStampedReference<>(100, 6666);

    public static void main(String[] args) throws InterruptedException {
        //线程1先修改为200,然后马上修改为100
        Thread thread1 = new Thread(() -> {
            System.out.println(Thread.currentThread().getName() + "第一次修改前时间戳版本号:" + count.getStamp());
            count.compareAndSet(100, 200, count.getStamp(), count.getStamp() + 1);
            System.out.println(Thread.currentThread().getName() + "第二次修改前时间戳版本号:" + count.getStamp());
            count.compareAndSet(200, 100, count.getStamp(), count.getStamp() + 1);
            System.out.println(Thread.currentThread().getName() + "完成对count的从100修改为200,然后再修改为100");
        });
        Thread thread2 = new Thread(() -> {
            try {
                Thread.sleep(5);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            System.out.println(Thread.currentThread().getName() + "第三次修改前时间戳版本号:" + count.getStamp());
            boolean isSuccess = count.compareAndSet(100, 300, count.getStamp(), count.getStamp() + 1);
            if (isSuccess) {
                System.out.println(Thread.currentThread().getName() + "修改成功,对count的从100修改为300");
            } else {
                System.out.println(Thread.currentThread().getName() + "修改失败,count值为:" + count.getReference());
            }

        });
        thread1.start();
        thread2.start();
    }
}

结果,因为时间戳已经被修改了,线程1就没法修改cout,避免了ABA问题:

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

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

相关文章

Rk3588 Android12 AIDL 开发

AIDL (Android Interface Definition Language) 和 HIDL (HAL Interface Definition Language) 都是 Android 系统中用于定义接口的工具&#xff0c;但它们有不同的用途和特性。 AIDL (Android Interface Definition Language) 用途&#xff1a; 主要用于应用程序之间的进程间…

基于SpringBoot的智能物流仓库管理系统-源代码-论文

&#x1f4a5;&#x1f4a5;源码和论文下载&#x1f4a5;&#x1f4a5;&#xff1a;基于SpringBoot的智能物流仓库管理系统-源代码-论文-数据库 1. 系统介绍 物流快递仓库管理是一项非常繁琐复杂的工作&#xff0c;每天要处理大量的单据数据&#xff0c;包括入库、出库、退库、…

Web:攻防世界unseping

目录 一、初见 二、解题步骤 一、初见 刚拿到这个题目&#xff0c;可以发现是一串PHP代码&#xff0c;关注到unserialize(base64_decode($ctf))函数&#xff0c;就知道是考php反序列化。 审计PHP类ease&#xff1a; __construct($method, $args) &#xff1a;构造器需要传入两…

k8s中的层级结构,及节点组件的作用

可以从三个层面来看待k8s集群 第一个层 是client主机 为什么要有client主机 不是在控制节点上管理计算节点吗 是的 但是&#xff0c;当有多个k8s集群需要管理的时候 一般不会每管理一个集群&#xff0c;就登录到那个集群的控制节点上 这样来来回回切换登录 client主机…

电路分析 ---- 加法器

1 同相加法器 分析过程 虚短&#xff1a; u u − R G R G R F u O u_{}u_{-}\cfrac{R_{G}}{R_{G}R_{F}}u_{O} u​u−​RG​RF​RG​​uO​ i 1 u I 1 − u R 1 i_{1}\cfrac{u_{I1}-u_{}}{R_{1}} i1​R1​uI1​−u​​&#xff1b; i 2 u I 2 − u R 2 i_{2}\cfrac{u_{…

同时播放多个视频

介绍一款小众的视频播放器&#xff0c;之前有小伙伴找那种可以同时播放多个视频的软件&#xff0c;“恒硕加播放”可以做到这一点&#xff0c;功能不是太多&#xff0c;但是日常播放是足够了。 同时播放多个视频控制多个视频跳到指定进度同时暂停/播放/停止/静音/倍速浏览系统…

金属3D打印与压铸模具的融合:创新引领制造新纪元

步入工业4.0的新纪元&#xff0c;3D打印技术犹如一股强劲的东风&#xff0c;为制造业的转型升级插上了智慧的翅膀。作为铸造领域的核心&#xff0c;压铸模具在工业领域扮演着不可或缺的角色&#xff0c;其重要性不言而喻。然而&#xff0c;面对日益多样化的压铸件需求&#xff…

力扣最热一百题——和为K的子数组

目录 题目链接&#xff1a;560. 和为 K 的子数组 - 力扣&#xff08;LeetCode&#xff09; 题目描述 示例 提示&#xff1a; 解法一&#xff1a;暴力枚举 Java写法&#xff1a; C写法&#xff1a; 解法二&#xff1a;前缀和哈希表 计算子数组和 如何优化问题 代码解…

混合部署 | 在RK3568上同时部署RT-Thread和Linux系统-迅为电子

RT-Thread 是一个高安全性、实时性的操作系统&#xff0c;广泛应用于任务关键领域&#xff0c;例如电力、轨道交通、车载系统、工业控制和新能源等。它的加入让 RK3568 能够在保证系统实时性和安全性的同时&#xff0c;灵活处理复杂的任务场景。 在一般情况下&#xff0c;iTOP-…

AI聊天应用不能上架?Google play对AI类型应用的规则要求是什么?

随着生成式AI模型的广泛应用&#xff0c;很多开发者都有在开发AI应用或将其整合到应用中。我们知道&#xff0c;谷歌是非常注重应用生态的&#xff0c;去年开始就推出了一些针对生成式AI应用的政策&#xff0c;对AI应用的内容质量和合规性问题提出了一些要求。 几天前&#xff…

Deep Ocr

1.圈出内容,文本那里要有内容.然后你保存,并导出数据集. 2.找出deep_ocr_recognition_training_workflow.hdev 文件.修改“DatasetFilename : Test.hdict” 310行 write_deep_ocr (DeepOcrHandle, BestModelDeepOCRFilename) 3.推理test.hdev 但发现很慢&#xff0c;没有mlp…

STM32快速复习(九)RTC时钟模块

文章目录 前言一、RTC是什么&#xff1f;RTC的工作原理&#xff1f;二、库函数以及示例1.标准库函数2.示例代码 总结 前言 STM32 的实时时钟&#xff08;RTC&#xff09;是一个独立的定时器。 STM32 的 RTC 模块拥有一组连续计数的计数器&#xff0c;在相应软件配置下&#xf…

ARM----时钟

时钟频率可以是由晶振提供的,我们需要高频率,但是外部接高的晶振会不稳定,所有使用PLL(锁相环)来放大频率。接下来就让我们学习用外部晶振提供的频率来配置时钟频率。 一.时钟源的选择 在这里我们选择外部晶振作为时钟源,通过查看芯片手册和原理图来看我们的时钟源。 这是…

2024高教社杯全国大学生数学建模竞赛C题解析 | 思路 代码 论文

C题 农作物种植策略 完整论文模型的建立与求解数据清洗问题一的建模与求解问题二的建模与求解问题三的建模与求解 代码第一问 完整论文 本题是一个运筹优化问题。 对于第一问&#xff0c;题目要求在假定各种农作物未来的预期销售量、种植成本、亩产量和销售价格相对于2023年保…

信号的捕捉处理

文章目录 4 信号的捕捉处理4.1 内核如何实现信号的捕捉4.2 sigaction4.2.1 使用这个函数对2号信号进行捕捉4.2.2 pending位图什么时候由1变04.2.3 不允许信号重复发送 5. 其他5.1 可重入函数5.2 volatile5.3 SIGCHLD信号5.4 信号生命周期 4 信号的捕捉处理 4.1 内核如何实现信…

鸿蒙(API 12 Beta6版)图形加速【Vulkan平台】超帧功能开发

业务流程 基于Vulkan图形API平台&#xff0c;集成超帧内插模式的主要业务流程如下&#xff1a; 用户进入超帧适用的游戏场景。游戏应用调用[HMS_FG_CreateContext_VK]接口创建超帧上下文实例。游戏应用调用接口配置超帧实例属性。包括调用[HMS_FG_SetAlgorithmMode_VK]&#x…

第T11周:使用TensorFlow进行优化器对比实验

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 文章目录 一、前期工作1.设置GPU&#xff08;如果使用的是CPU可以忽略这步&#xff09; 二、导入数据1、导入数据2、检查数据3、配置数据集4、数据可视化 三、…

淘宝和微信支付“好”上了,打翻了支付宝的“醋坛子”?

文&#xff1a;互联网江湖 作者&#xff1a;刘致呈 最近&#xff0c;淘宝将全面接入微信支付的消息&#xff0c;在整个互联网圈里炸开了锅。 虽说阿里系平台与腾讯之间“拆墙”的消息&#xff0c;早就不算是啥新鲜事了。而且进一步互联互通&#xff0c;无论是对广大用户&…

43. 1 ~ n 整数中 1 出现的次数【难】

comments: true difficulty: 中等 edit_url: https://github.com/doocs/leetcode/edit/main/lcof/%E9%9D%A2%E8%AF%95%E9%A2%9843.%201%EF%BD%9En%E6%95%B4%E6%95%B0%E4%B8%AD1%E5%87%BA%E7%8E%B0%E7%9A%84%E6%AC%A1%E6%95%B0/README.md 面试题 43. 1 &#xff5e; n 整数中 1 …

(postman)接口测试进阶实战

1.内置和自定义的动态参数 内置的动态参数有哪些&#xff1f; ---{{$}}--是内置动态参数的标志 //自定义的动态参数 此处date.now()的作用就相当于上面的timestamp 2.业务闭环及文件接口测试 返回的url地址可以在网页中查询得到。 3. 常规断言&#xff0c;动态参数断言&#xf…