并发编程之CAS

写在前面

上一篇我们分析了volatile变量对于内存可见性的保证以及抑制指令重排的特性,了解到在多线程对volatile变量的读写不会发生线程阻塞,但是volatile变量只能保证可见性和有序性,并不能保证原子性,即i++之后的数据仍然是不正确的。目前保证原子性的方式只有加锁,无论是内置锁(synchronized),还是显示锁(ReentrantLock),加锁意味着线程阻塞,如果竞争激烈,很可能导致频繁的线程上下文切换,从而大大降低了性能。鉴于此,我们希望有一种机制,既可以保证原子性,又能够使得线程不发生阻塞,幸运的是,java的先行者已经为我们做好了这些,这就是今天的主角——CAS。

CAS分析

CAS,即compare and swap,比较更新机制。它涉及到三个参数,内存值V,旧值A,新值B。当且仅当V=A时,我们才去将变量的值更新成B并返回true,否则返回false。(这个比较再更新的原子性是由操作系统底层来实现的,我们并不需要关心)

JUC下的atomic类都是通过CAS来实现的,我们以AtomicInteger为例来说明。

AtomicInteger i = new AtomicInteger(0);
i.increasementAndGet();
01.png

如图所示,我们开启了三个线程并发对AtomicInteger类型的变量i进行increasementAndGet的自增操作,i的值初始化为0,首先,每个线程都会先去读取i的值,然后进行院子的CAS操作,在CAS操作里,如果此时i的值还是之前读到的值,则更新为新值;如果CAS操作里i的值不少刚刚获取的值,说明有其他线程先于本线程修改了i,CAS失败,失败之后自旋,再次读取i的值,再次CAS操作。

对应于上图,我们假设线程1先于线程2,3执行,线程2,3同时执行,完整的流程就是:

  • 线程1首先拿到变量id值为0,此时就它一个线程,直接进行CAS操作,将0改成1
  • 线程2,线程3同时读取到变量id值为1,假设线程2抢先一步发起CAS操作,在CAS操作里,它会检查此时变量i的值是否还是1,如果是,则更新为2。
  • 线程3发起CAS操作,它去检查变量i的值,发现不是1而是2,CAS失败。然后进入自旋,再次读取变量i的值为2,紧接着进入CAS操作,检查变量i是否还是2,此时是2,则更新为3。

上述过程,就是Atomic原子类的实现原理,并没有基于加锁串行化处理,通过CAS方式,无锁地进行更新,而CAS是由操作系统底层保证了其操作的原子性。

CAS实现原子操作的三大问题

  1. ABA问题
    CAS操作需要检查当前内存的变量值是否和刚刚读取的值相同,假设一个变量是A,变成了B,之后又变成了A,所以在进行CAS操作的时候检查到它的值没有发生变化。ABA的解决方案就是加上版本号1A->2B->3A,类似这种。java先行者们也给我们提供了AtomicStampedReference来解决ABA的问题,其中的预期标志也类似于版本号的功能。
  2. 循环时间长开销大
    多线程竞争激烈的情况下进行CAS操作,会导致某些线程长时间空循环,也就是说它什么都没做,只是不停地在浪费处理器的处理时间而已。
  3. 只能保证一个共享变量的原子操作
    一个共享变量的操作可以用CAS保证其原子性,多个共享变量的操作,循环CAS是无法保证其原子性的。有个取巧的办法,java先行者设计的AtomicReference类,我们可以通过将多个变量放到一个对象里面,然后由AtomicReference进行原子性地更新。

参考文档

方腾飞《并发编程的艺术》

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

推荐阅读更多精彩内容