Java并发编程 CAS 详解

一. 书面概述

CAS的全称为Compare And Swap,直译就是比较交换。是一条CPU的原子指令,其作用是让CPU先进行比较两个值是否相等,然后原子地更新某个位置的值,其实现方式是基于硬件平台的汇编指令,在intel的CPU中,使用的是cmpxchg指令,就是说CAS是靠硬件实现的,从而在硬件层面提升效率。

CAS有三个操作数:内存值V、旧的预期值A、要修改的值B,当且仅当预期值A和内存值V相同时,将内存值修改为B并返回true,否则什么都不做并返回false。

二. sun.misc.Unsafe介绍

工欲善其事必先利其器,为什么要先讲Unsafe?

Unsafe类是进行底层操作的方法集合,可以直接操作内存,进行一些非常规操作,所以说是"不安全"的操作,但是因为直接操作内存,它的效率很高,通常在在对性能有要求或者有底层操作需求的时候使用。

我们的CAS操作就是通过sun.misc.Unsafe类操作的(java8以下),Unsafe在jdk1.8.0/jre/lib/rt.jar包下。

怎么获取Unsafe实例?

public final class Unsafe {
    private static final Unsafe theUnsafe;
    private Unsafe() {}
    static{
         theUnsafe = new Unsafe();
    }
    public static Unsafe getUnsafe() {
         return theUnsafe;
    }
}

这里我们没法直接new对象,必须要通过反射来获取 theUnsafe 变量,下面来看下里面的几个重要方法

  1. public long objectFieldOffset(Field f)
    获取字段的内存偏移地址,cas要用。内部是native代码实现的,不讲, 看一段实例代码:
 private static Object unsafe;
    static {
        try {
            /** Unsafe在rt.jar下,不能直接实例化。必须通过反射 */
            Field field = Class.forName("sun.misc.Unsafe").getDeclaredField("theUnsafe");
            field.setAccessible(true);
            unsafe = field.get(null);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    static class Data {
        int intParam;
    }
    public static void main(String[] args) throws Exception {
        // 反射获取objectFieldOffset方法
        Method method = Class.forName("sun.misc.Unsafe").getDeclaredMethod("objectFieldOffset", new Class<?>[] {Field.class});
        method.setAccessible(true);
        // 执行调用, 返回 Data类的intParam成员的偏移地址
        Object ret = method.invoke(unsafe, Data.class.getDeclaredField("intParam"));
        System.err.println(ret);
}

打印结果 : 12

static静态块就是取得了Unsafe 类中的单例theUnsafe ,然后反射调用其objectFieldOffset方法,返回对象成员的内存偏移量。

  1. public final native boolean compareAndSwapInt(Object o, long offset, int expected, int x);
    这是重头戏,CAS操作的方法实现, 将对象o的偏移地址变量改成x,前提是x的值是expected,请接着上面的代码:
    public static void main(String[] args) throws Exception {
        // 反射获取objectFieldOffset方法
        Method method = Class.forName("sun.misc.Unsafe").getDeclaredMethod("objectFieldOffset", new Class<?>[] {Field.class});
        method.setAccessible(true);
        // 执行调用, 返回 Data类的intParam成员的偏移地址
        long offset = (long) method.invoke(unsafe, Data.class.getDeclaredField("intParam"));
        // 获取 compareAndSwapInt 方法
        method = Class.forName("sun.misc.Unsafe").getDeclaredMethod("compareAndSwapInt", new Class<?>[] {Object.class,
            long.class,int.class,int.class});
        method.setAccessible(true);
        Data data = new Data();
        data.intParam = 78;
        // 第4个参数: 预期的值   第5个参数: 要修改的值
        boolean success = (boolean) method.invoke(unsafe, data,offset,7,90);
        System.err.println("1 修改成功吗:"+success+ " , 修改后intParam:"+data.intParam);
        success = (boolean) method.invoke(unsafe, data,offset,78,90);
        System.err.println("2 修改成功吗:"+success+ " , 修改后intParam:"+data.intParam);
    }

我们发现第一次 预期值传了7 (实际上是78),所以我们修改失败,第二次才成功。

  1. public native int getIntVolatile(Object o, long offset);
    获得给定对象的指定偏移量offset的int值,使用volatile语义,总能获取到最新的int值。就是获取的主内存的值,并不是自己线程的副本。
    我们都知道JMM内存模型 ,线程自己内存拥有一套副本,和主内存不一致 ,所以一个线程操作一个变量,另一个线程自己的副本不一定马上会更新,这样就会导致线程安全。

请看用CAS是如何解决上述问题的

    public static void main(String[] args) throws Exception {
        // 反射获取objectFieldOffset方法
        Method method = Class.forName("sun.misc.Unsafe").getDeclaredMethod("objectFieldOffset", new Class<?>[] {Field.class});
        method.setAccessible(true);
        // 执行调用, 返回 Data类的intParam成员的偏移地址
        long offset = (long) method.invoke(unsafe, Data.class.getDeclaredField("intParam"));
        // 获取 compareAndSwapInt 方法
        method = Class.forName("sun.misc.Unsafe").getDeclaredMethod("compareAndSwapInt", new Class<?>[] {Object.class,
            long.class,int.class,int.class});
        method.setAccessible(true);
        Data data = new Data();
        data.intParam = 78;
        
        while(true) {
            method = Class.forName("sun.misc.Unsafe").getDeclaredMethod("getIntVolatile", new Class<?>[] {Object.class,
                long.class});
            //通过 getIntVolatile 方法获取主内存的值
            int expected = (int) method.invoke(unsafe, data,offset);
            // 比较主内存的值 和当前 线程副本的值是否一致,一致就更新,否则更新失败, 
            method = Class.forName("sun.misc.Unsafe").getDeclaredMethod("compareAndSwapInt", new Class<?>[] {Object.class,
                long.class,int.class,int.class});
            boolean success = (boolean) method.invoke(unsafe, data,offset,expected,90);
            System.err.println(success);
            if(success) {
                break;
            }
            // 更新失败,循环重试,直到更新成功为止
        }
    }

借助了 getIntVolatile 先获取主内存的值, 然后compareAndSwapInt 将值一直循环更新成功为止。这其实也就是我们所说的自旋锁

其实java并发编程里面的juc包下的,什么AQS啊,AtomicInteger 等都是以上面这种骚操 作基础的,下面我们看下AtomicInteger 如何骚 的。

三. AtomicInteger 源码分析

public class AtomicInteger extends Number implements java.io.Serializable {
  private static final Unsafe unsafe = Unsafe.getUnsafe();
    private static final long valueOffset;
    static {
        try {
            valueOffset = unsafe.objectFieldOffset
                (AtomicInteger.class.getDeclaredField("value"));
        } catch (Exception ex) { throw new Error(ex); }
    }
    private volatile int value;
}

jdk当然可以直接使用getUnsafe方法来获取实例,然后把value的内存偏移量存储到valueOffset变量上,后面CAS操作直接用。value 就是AtomicInteger 实际存储的值。且是 volatile 的

incrementAndGet方法

   public final int incrementAndGet() {
        return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
    }

 public final int getAndAddInt(Object var1, long var2, int var4) {
   int var5;
   do {
   // 获取主内存的值
      var5 = this.getIntVolatile(var1, var2);
    // 将值变成原有的值var5 加上var4
   } while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));
   return var5;
}

直接调用Unsafe的getAndAddInt方法。getAndAddInt在多线程下也是安全的。

get方法

  public final int get() {
        return value;
    }

总结一下

(1)AtomicInteger中维护了一个使用volatile修饰的变量value,保证可见性;
(2)AtomicInteger中的主要方法最终几乎都会调用到Unsafe的compareAndSwapInt()方法保证对变量修改的原子性。

三. CAS总结

  • CAS机制只能保证共享变量操作的原子性,而不能保证代码块的原子性。

  • CAS操作就是基于处理器的CMPXCHG汇编指令实现的,因此,JVM中的CAS的原子性是处理器保障的。CAS是一种乐观锁的思想。

  • CAS自旋锁意思: 发现线程自己内存副本和主内存不一致(代表有多线程在竞争操作)就返回修改失败,然后循环CAS直到修改成功。

  • CAS解决的问题是: 不加锁确保某一变量的操作没有被其他线程修改过。

四. CAS带来的问题

1. ABA问题

假如你很牛逼,扣款的代码直接不加锁而是使用CAS来写。有这样一个场景:

  • A账户上有10块钱,娶媳妇需要提款5元,但是系统问题同时发起了两次扣款,相当于2个线程1,2并发。
  • 假如线程1先执行CAS,预期值是10,要修改成5 ,成功。然后准备到线程2,正常情况是 线程2 发现预期值是10,现在是5了,就会CAS失败不扣钱,这样系统就不会扣两次钱没问题, 但是发生了下面情况。
  • 在线程2 CAS之前,A的妈妈怕儿子娶媳妇钱不够,又往A账户上打了5块钱,这时,A的账户就恢复了10块钱。
  • 然后线程2 CAS 发现 卧槽,预期值是10,现在也是10,就毫不犹豫把钱扣了。A又只剩5块了。

妈妈,五块钱没了,我不取媳妇了,呜呜~~~~~~
其实上述问题原因就是CAS操作将值由A改为B然后又改成A , 另一个线程CAS的话是当做什么都没发生的。

看下JDK怎么利用 AtomicStampedReference 来解决这个问题的

public class AtomicStampedReference<V> {
    private static class Pair<T> {
        final T reference;
        final int stamp;
        private Pair(T reference, int stamp) {
            this.reference = reference;
            this.stamp = stamp;
        }
        static <T> Pair<T> of(T reference, int stamp) {
            return new Pair<T>(reference, stamp);
        }
    }
    private volatile Pair<V> pair;
    public boolean compareAndSet(V   expectedReference,
            V   newReference,
            int expectedStamp,
            int newStamp) {
        // 获取当前的(元素值,版本号)对
        Pair<V> current = pair;
        return
        // 引用没变
        expectedReference == current.reference &&
        // 版本号没变
        expectedStamp == current.stamp &&
        // 新引用等于旧引用
        ((newReference == current.reference &&
        // 新版本号等于旧版本号
        newStamp == current.stamp) ||
        // 构造新的Pair对象并CAS更新
        casPair(current, Pair.of(newReference, newStamp)));
        }
        
        private boolean casPair(Pair<V> cmp, Pair<V> val) {
        // 调用Unsafe的compareAndSwapObject()方法CAS更新pair的引用为新引用
        return UNSAFE.compareAndSwapObject(this, pairOffset, cmp, val);
    }
}
  • 首先把我们上面CAS操作的int,变成CAS操作对象Pair,原理是一样。
  • 加了个版本号stamp,只有版本号不一样时,CAS才操作成功。
  • 上面代码流程: 如果元素值和版本号都没有变化,并且和新的也相同,返回true;如果元素值和版本号都没有变化,并且和新的不完全相同,就构造一个新的Pair对象并执行CAS更新pair。

2. 并发自旋耗cpu多
在并发量比较低的情况下,线程冲突的概率比较小,自旋的次数不会很多。但是,高并发情况下,N个线程同时进行自旋操作,会出现大量失败并不断自旋的场景。 JDK8中出现了 LongAdder 来解决AtomicLong的上述并发大的问题。

AtomicLong中有个内部变量value保存着实际的long的值,高并发场景下,value变量就是N个线程竞争的一个热点。

LongAdder的基本思路就是分散热点,将value值分散到一个数组中,不同线程会命中到数组的不同槽中,各个线程只对自己槽中的那个值进行CAS操作,这样热点就被分散了,冲突的概率就小很多。如果要获取真正的long值,只要将各个槽中的变量值累加返回即可。

CAS就讲到这里吧~ 写东西太累了,还特别花时间。这些都是上班时间写的。

《 合抱之木,生于毫末;九层之台,起于累土;千里之行,始于足下 》
释义:合抱的大树,生长于细小的幼苗;九层的高台,筑起于每一堆泥土;千里的远行,是从脚下第一步开始走出来的

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容