CAS你知道吗?如何实现?

1. compareAndSet

volatile当中我们提到,volatile不能保证原子语义,所以当用到变量自增时,如果用到synchronized会太”重“了,在多线程环境下我们一般用原子类如AtomicInteger,其底层是CAS,volatile见此篇

  1. public final boolean compareAndSet(int expect, int update) {
  2. return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
  3. }

上述代码表示:

  • 如果线程的期望值和物理内存的真实值一样,那么就修改为更新值
  • 如果不一样,本次修改失败,就需要重新获取主物理内存的值

简单的代码例子:

  1. package com.yuxue.juc.CASTest;
  2. import java.util.concurrent.atomic.AtomicInteger;
  3. /**
  4. * CAS:比较并交换
  5. */
  6. public class CASDemo {
  7. public static void main(String[] args) {
  8. AtomicInteger atomicInteger = new AtomicInteger();
  9. //compareAndSet返回的是boolean类型,修改成功返回true,失败返回false
  10. System.out.println(atomicInteger.compareAndSet(0, 666) + "\t current data is " + atomicInteger.get());
  11. System.out.println(atomicInteger.compareAndSet(1, 777) + "\t current data is " + atomicInteger.get());
  12. System.out.println(atomicInteger.compareAndSet(666, 888) + "\t current data is " + atomicInteger.get());
  13. }
  14. }

输出为:

  1. //第一次期望值是0,原值默认0,所以CAS成功修改为666
  2. true current data is 666
  3. //第二次期望值是1,原值第一步修改为666,所以CAS不成功修改
  4. false current data is 666
  5. //第三次期望值是666,原值第一步修改为666,所以CAS成功修改为888
  6. true current data is 888

2. CAS底层原理?对Unsafe的理解

我们都知道,atomicInteger.getAndIncrement()方法能够在多线程环境下保证变量的安全同时让其自增,但是源码当中也没有synchronized,那么如何保证底层安全?如果保证多线程环境下的变量安全?我们打开其源码:

2.1 compareAndSet

  1. public final boolean compareAndSet(int expect, int update) {
  2. return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
  3. }

里面有用到unsafe对象的compareAndSwapInt方法,再找unsafe源码

其底层用到Unsafe类来保证线程安全!

2.2 Unsafe

  • 是CAS核心类,由于Java方法无法直接访问地层系统,需要通过本地(native)方法来访问,Unsafe相当 于一个后门,基于该类可以直接操作特定内存数据。Unsafe类存在于sun.misc 包中,其内部方法操作可 以像C的指针一样直接操作内存,因为Java中CAS操作的执行依赖于Unsafe类的方法。

  • Unsafe类中的所有方法都是native修饰的,也就是说Unsafe类中的方法都直接调用操作系统底层资源执行相应任务

  • 变量valueOffset,表示该变量值在内存中的偏移地址,因为Unsafe就是根据内存偏移地址获取数据的

  • 变量value用volatile修饰,保证多线程之间的可见性

2.3 CAS是什么

CAS全称呼Compare-And-Swap,它是一条CPU并发原语

  • 他的功能是判断内存某个位置的值是否为预期值,如果是则更改为新的值,这个过程是原子的
  • CAS并发原语体现在JAVA语言中就是sun.misc.Unsafe类中各个方法。调用Unsafe类中的CAS方法,JVM会帮我们实现CAS汇编指令。这是一种完全依赖于硬件的功能,通过他实现了原子操作。
  • 由于CAS是一种系统原语,原语属于操作系统用语范畴,是由若干条指令组成的,用于完成某个功能的一个过程,并且原语的执行必须是连续的,在执行过程中不允许被中断,也就是说CAS是一条CPU的原子指令,不会造成数据不一致问题

所以!CAS是通过Unsafe类的CPU指令源语来保证数据的原子性!

CAS源码:

  1. //unsafe.getAndAddInt
  2. //var1对应
  3. public final int getAndAddInt(Object o, long offset, int delta) {
  4. int v;
  5. do {
  6. v = getIntVolatile(o, offset);
  7. } while (!compareAndSwapInt(o, offset, v, v + delta));
  8. return v;
  9. }

o this即AtomicInteger对象本身

offset 该对象的引用地址(偏移地址)

delta 需要增加的变量

v通过AtomicInteger对象本身的offset偏移地址找出的主内存中真实的值,用该对象前的值与v比较; 如果相同,更新v+delta并且返回true, 如果不同,继续去之然后再比较,直到更新完成

2.4 总结

CAS:比较当前工作内存中的值和主物理内存中的值,如果相同则执行规定操作,否则的话继续比较直到主内存和工作内存中的值一值!(不清楚工作内存以及主内存的请移步查看volatile中的JMM模型

3. CAS缺点

3.1 循环时间长,开销大

例如getAndAddInt方法执行,有个do...while循环,如果CAS失败,一直会进行尝试,如果CAS长时间不成功, 可能会给CPU带来很大的开销(自旋!)

3.2 只能保证一个共享变量的原子操作

对多个共享变量操作时,循环CAS就无法保证操作的原子性,这个时候就可以用锁来保证原子性

3.3 ABA问题

CAS算法实现一个重要前提需要去除内存中某个时刻的数据并在当下时刻比较并替换,那么在这个时间差类会导致数据的变化

例:比如线程1从内存位置V取出A,线程2同时也从内存取出A,并且线程2进行一些操作将值改为B,然后线程2又将V位置数据改成A,这时候线程1进行CAS操作发现内存中的值依然时A,然后线程1操作成功

尽管线程1的CAS操作成功,但是不代表这个过程没有问题

4. 原子类AtomicInteger的ABA问题?原子更新引用?

首先我们已经阐述了ABA的概念以及问题,首先我们要知道原子引用类的概念

4.1 原子引用

  1. AtomicReference<V>

这里的V只要是其他的类均可使用AtomicReference作为其包装类

示例代码:

  1. package com.yuxue.juc.CASTest;
  2. import lombok.AllArgsConstructor;
  3. import lombok.Data;
  4. import lombok.NoArgsConstructor;
  5. import java.util.concurrent.atomic.AtomicReference;
  6. @Data
  7. @NoArgsConstructor
  8. @AllArgsConstructor
  9. class User{
  10. private String name;
  11. private int age;
  12. }
  13. /**
  14. * 测试原子引用类
  15. * */
  16. public class AtomicReferenceTest {
  17. public static void main(String[] args) {
  18. User u1 = new User("张三",18);
  19. User u2 = new User("李四",23);
  20. AtomicReference<User> atomicReference = new AtomicReference<>();
  21. atomicReference.set(u1);
  22. System.out.println(atomicReference.compareAndSet(u1, u2) + "\t" + atomicReference.get().toString());
  23. System.out.println(atomicReference.compareAndSet(u1, u2) + "\t" + atomicReference.get().toString());
  24. }
  25. }

输出结果为:

  1. true User(name=李四, age=23)
  2. false User(name=李四, age=23)

4.2 ABA问题代码实现

解决方案:带时间戳的原子引用

首先ABA问题代码展示:

  1. package com.yuxue.juc.CASTest;
  2. import java.util.concurrent.atomic.AtomicReference;
  3. import java.util.concurrent.atomic.AtomicStampedReference;
  4. public class ABASolution {
  5. static AtomicReference<Integer> atomicReference = new AtomicReference<>(100);
  6. public static void main(String[] args) {
  7. new Thread(() -> {
  8. //ABA问题
  9. System.out.println(atomicReference.compareAndSet(100, 101) + "\t" + Thread.currentThread().getName() + "value is:" + atomicReference.get());
  10. System.out.println(atomicReference.compareAndSet(101, 100) + "\t" + Thread.currentThread().getName() + "value is:" + atomicReference.get());
  11. }, "t1").start();
  12. new Thread(() -> {
  13. //先休眠,让t1线程完成ABA操作
  14. try {
  15. Thread.sleep(1000);
  16. System.out.println(atomicReference.compareAndSet(100, 2019) + "\t" + Thread.currentThread().getName() + "value is:" + atomicReference.get());
  17. } catch (InterruptedException e) {
  18. e.printStackTrace();
  19. }
  20. }, "t2").start();
  21. }
  22. }

结果为:

  1. true t1 value is:101
  2. true t1 value is:100
  3. true t2 value is:2019

可以看到t1首先修改值为101,之后又修改回来100,但是线程t2的工作内存中还是100,之后与主内存相比,发现主内存值也是100,之后放心修改值为2019,此时就会出现ABA问题

4.3 所以?怎么解决ABA问题?

采用内置的类AtomicStampedReference<V>其为携带时间戳的类,我们可以每次更改值时对时间戳进行操作,这样就可以保证不会出现ABA问题

  1. package com.yuxue.juc.CASTest;
  2. import java.util.concurrent.atomic.AtomicReference;
  3. import java.util.concurrent.atomic.AtomicStampedReference;
  4. public class ABASolution {
  5. //创建变量,第一个是initialRef为初始值,第二个是initialStamp为初始化时间戳
  6. static AtomicStampedReference<Integer> atomicStampedReference = new AtomicStampedReference<>(100, 1);
  7. public static void main(String[] args) {
  8. //atmoinReferenceMethod();
  9. new Thread(() -> {
  10. //获得时间戳,此时为1
  11. int stamp = atomicStampedReference.getStamp();
  12. System.out.println(Thread.currentThread().getName() + "\t第1次版本号" + stamp);
  13. try {
  14. //此处休眠的目的是为了让t2获得初始版本号
  15. Thread.sleep(1000);
  16. //第一次修改,值改为101,版本号加1
  17. atomicStampedReference.compareAndSet(100, 101, atomicStampedReference.getStamp(), atomicStampedReference.getStamp() + 1);
  18. System.out.println(Thread.currentThread().getName() + "\t第2次版本号" + atomicStampedReference.getStamp());
  19. //第二次修改,值改为100,版本号加1
  20. atomicStampedReference.compareAndSet(101, 100, atomicStampedReference.getStamp(), atomicStampedReference.getStamp() + 1);
  21. System.out.println(Thread.currentThread().getName() + "\t第3次版本号" + atomicStampedReference.getStamp());
  22. } catch (InterruptedException e) {
  23. e.printStackTrace();
  24. }
  25. }, "t1").start();
  26. new Thread(() -> {
  27. //获得时间戳,此时为1
  28. int stamp = atomicStampedReference.getStamp();
  29. System.out.println(Thread.currentThread().getName() + "\t第1次版本号" + stamp);
  30. try {
  31. //休眠2秒,此时线程t1已经将值改变但是又变回来,为ABA问题
  32. Thread.sleep(2000);
  33. //首先尝试是否可以根据值和时间戳进行更改
  34. boolean result = atomicStampedReference.compareAndSet(100, 2019, stamp, stamp + 1);
  35. System.out.println(Thread.currentThread().getName() + "\t修改是否成功" + result + "\t当前最新实际版本号" + atomicStampedReference.getStamp());
  36. System.out.println(Thread.currentThread().getName() + "\t当前最新实际值" + atomicStampedReference.getReference());
  37. } catch (InterruptedException e) {
  38. e.printStackTrace();
  39. }
  40. }, "t2").start();
  41. }
  42. }

输出结果:

  1. t1 1次版本号1
  2. t2 1次版本号1
  3. //t1修改两次,版本号加了2
  4. t1 2次版本号2
  5. t1 3次版本号3
  6. //t2判断版本号,之后再决定能不能改
  7. t2 修改是否成功false 当前最新实际版本号3
  8. //实际并没有进行更改
  9. t2 当前最新实际值100

这样,我们用JUC内置atomic下的AtomicStampedReference类来解决了ABA问题

CAS你知道吗?底层如何实现?ABA问题又是什么?关于这些你知道答案吗的更多相关文章

  1. CAS底层原理与ABA问题

    CAS定义 CAS(Compare And Swap)是一种无锁算法.CAS算法是乐观锁的一种实现.CAS有3个操作数,内存值V,旧的预期值A,要修改的新值B.当预期值A和内存值V相同时,将内存值V修 ...

  2. java高并发核心要点|系列3|锁的底层实现原理|ABA问题

    继续讲CAS算法,上篇文章我们知道,CAS算法底层实现,是通过CPU的原子指令来实现. 那么这里又有一个情景: 话说,有一个线程one从内存位置V中取出A,这时候另一个线程two也从内存中取出A,并且 ...

  3. 基础篇:详解锁原理,volatile+cas、synchronized的底层实现

    目录 1 锁的分类 2 synchronized底层原理 3 Object的wait和notify方法原理 4 jvm对synchronized的优化 5 CAS的底层原理 6 CAS同步操作的问题 ...

  4. CAS核心思想、底层实现

    ★ 1.CAS 是什么 CAS 是比较并交换,是实现并发算法时常用到的一种技术.当内存的值和期望的值相等时,进行更新,否则 什么都不做 或 重来 . CAS 的底层实现:是靠硬件实现的,靠硬件的原子性 ...

  5. 沉淀再出发:java中的CAS和ABA问题整理

    沉淀再出发:java中的CAS和ABA问题整理 一.前言 在多并发程序设计之中,我们不得不面对并发.互斥.竞争.死锁.资源抢占等等问题,归根到底就是读写的问题,有了读写才有了增删改查,才有了所有的一切 ...

  6. 非阻塞式的原子性操作-CAS应用及原理

    一:问题抛出 假设在出现高并发的情况下对一个整数变量做依次递增操作,下面这两段代码是否会出现问题? 1. public class IntegerTest { private static Integ ...

  7. CAS 无锁式同步机制

    计算机系统中,CPU 和内存之间是通过总线进行通信的,当某个线程占有 CPU 执行指令的时候,会尽可能的将一些需要从内存中访问的变量缓存在自己的高速缓存区中,而修改也不会立即映射到内存. 而此时,其他 ...

  8. 详解java中CAS机制所导致的问题以及解决——内存顺序冲突

    [CAS机制] 指的是CompareAndSwap或CompareAndSet,是一个原子操作,实现此机制的原子类记录着当前值的在内存中存储的偏移地址,将内存中的真实值V与旧的预期值A做比较,如果不一 ...

  9. 无锁的同步策略——CAS操作详解

    目录 1. 从乐观锁和悲观锁谈起 2. CAS详解 2.1 CAS指令 2.3 Java中的CAS指令 2.4 CAS结合失败重试机制进行并发控制 3. CAS操作的优势和劣势 3.1 CAS相比独占 ...

随机推荐

  1. [Java] HOW2J(Java初级)

    变量 基本类型:整型(byte.short.int.long).字符型(char).浮点型(float.double).布尔型(boolean) 给基本类型赋值的方式叫字面值 字符的字面值放在单引号中 ...

  2. KVM 添加新硬件

    1 显卡 spice 2视频 qxl驱动 3 声音 ich6最好  ich9最清楚 4 输入 鼠标 智能图   否则不能VNC找不到焦点 5 磁盘大小 至少80G 否则 无法自动安装  无swap和 ...

  3. stressapptest测试用例testcase方法aarch64

    ### https://github.com/stressapptest/stressapptest aarch64 To build from source, the build/installat ...

  4. IT菜鸟之OSI七层模型

    OSI七层模型从下到上分别是: 应用层 表示层 会话层 传输层 网络层 数据链路层 物理层 第一层物理层: 物理层是传输媒介(网线.无线.光纤) 在线路中起到的作用:是将0/1转换成电信号或光信号 物 ...

  5. shell基础之EOF的用法

    一.EOF的用法 EOF是(END Of File)的缩写,表示自定义终止符.既然自定义,那么EOF就不是固定的,可以随意设置别名,在linux按ctrl-d 就代表EOF. EOF一般会配合cat能 ...

  6. jenkins部署vue项目

    一.新建自由风格的项目 二.配置项目 三.部分部署脚本 #!/bin/bashecho $PATHnpm config set proxy nullnpm config set https-proxy ...

  7. php反转字符串的三种方法

    (假设有字符串abcd,用php实现字符串翻转) 1.第一种php有自带的函数strrev可以轻松实现: $str = 'abcd'; //第一种自带strrev实现翻转 echo strrev($s ...

  8. Dubbo以及Zookeeper安装

    1.什么是Dubbo? Apache Dubbo 是一款高性能.轻量级的开源 Java 服务框架 提供了六大核心能力:面向接口代理的高性能RPC调用,智能容错和负载均衡,服务自动注册和发现,高度可扩展 ...

  9. 用python调试Appium和雷电模拟器连接时出现Original error: Could not find 'adb.exe' in PATH

    用python调试Appium和雷电模拟器连接时出现Original error: Could not find 'adb.exe' in PATH 确定环境变量没错,用管理员启动Appium就不会出 ...

  10. 微信内 H5 页面自定义分享

    起源: 最近公司在做一个活动的h5页面,在微信内打开时需要进行微信授权,然后后端会重定向到这个页面并且携带了一些参数(openid等).问题是点击微信的原生分享时,会把携带的这些参数一起分享出去,等于 ...