CAS原理

CAS(compare-and-Swap)简介

比较替换,是一种实现并发算法的常用技术

其核心是Usafe类,内部的方法都是native方法,这些方法可以直接操作内存中的数据,访问系统底层。

public class AtomicInteger extends Number implements java.io.Serializable {
    private static final long serialVersionUID = 6214790243416807050L;

    // setup to use Unsafe.compareAndSwapInt for updates
    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); }
    }
//...

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

}

CAS底层原理

java.util.concurrent包完全建立在CAS上,CAS的核心是比较交换,首先给出一个期望值A,修改值B,然后比较当前内存的值V,当A==V,就赋值,否则就返回false。

Usafe底层是由内联函数(插入汇编)编写的,首先判断当前是否为多核处理器,如果是则加锁,比较操作为原子操作。

变量valueOffset,该变量在内存中的偏移地址,Unsafe通过内存偏移地址来获取数据

value使用volatile修饰,设置了多线程的可见性

分析一下自增代码,调用的最后还是compareAndSwapInt

public final int getAndIncrement() {
   return unsafe.getAndAddInt(this, valueOffset, 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;
   }
}

C++实现如下

UNSAFE_ENTRY(jboolean, Unsafe_CompareAndSwapInt(JNIEnv *env, jobject unsafe, jobject obj, jlong offset, jint e, jint x))
  UnsafeWrapper("Unsafe_CompareAndSwapInt");
  oop p = JNIHandles::resolve(obj);
  jint* addr = (jint *) index_oop_from_field_offset_long(p, offset);
  return (jint)(Atomic::cmpxchg(x, addr, e)) == e;
UNSAFE_END

unsigned Atomic::cmpxchg(unsigned int exchange_value,
                         volatile unsigned int* dest, unsigned int compare_value) {
  assert(sizeof(unsigned int) == sizeof(jint), "more work to do");
  return (unsigned int)Atomic::cmpxchg((jint)exchange_value, (volatile jint*)dest,
                                       (jint)compare_value);
}

linux实现

inline jint Atomic::cmpxchg(jint exchange_value, volatile jint* dest, jint compare_value) {
  int mp = os::is_MP();
  __asm__ volatile (LOCK_IF_MP(%4) "cmpxchgl %1,(%3)"
                    : "=a" (exchange_value)
                    : "r" (exchange_value), "a" (compare_value), "r" (dest), "r" (mp)
                    : "cc", "memory");
  return exchange_value;
}

CAS缺点

代码中存在循环(自旋锁),需要等待其他线程完成,不断重试,等到达到期望值再进行操作

每次只能保证一个变量的原子操作

ABA问题:自增操作为例1.获取地址V的变量A,2.根据A获取变量值B,3.V地址设置为B

如果这个操作过程中,初始变量为A,结果为B,但是过程中如果被其他线程操作过(例如变为B之后,又把地址变为A,下一次取得时候会认为这个值没有变化)

CAS通过版本标记来解决这个问题AtomicStampedReference

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值