前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >理解Java并发里面的CAS概念

理解Java并发里面的CAS概念

作者头像
我是攻城师
发布2018-07-23 10:53:16
9960
发布2018-07-23 10:53:16
举报
文章被收录于专栏:我是攻城师我是攻城师

前言

我们知道在Java多线程里面关于共享变量的操作,一定是要使用线程同步来保证线程安全的,一旦涉及线程同步,就需要加锁,一旦加锁就意味着某一个时候只能有一个线程在操作,其他的线程如果没有得到锁就会阻塞起来,此时的线程的状态是BLOCKED,当前面的线程释放锁的时候,系统会自动调度当前的线程进入临界区,这里面存在一个问题,就是线程的上下文切换的问题,虽然比起来进程的上下文切换,线程的上下文切换更轻量级,但仍然也是有一定开销的,比如最简单的i++的例子,那么如何有没有一种不需要加锁也能保证线程安全的数据结构呢?答案是肯定的,这就是今天需要谈到的CAS(Compare And Swap或 Compare And Set)。

什么是CAS

CAS通常是指Compare And Swap或 Compare And Set)是硬件操作系统级别提供的具有原子性的原语指令,利用它可以在多线程中取得和同步一样的效果。

CAS的原理

CAS 算法大致原理是:在对变量进行计算之前(如 ++ 操作),首先读取原变量值,称为 旧的预期值 A,然后在更新之前再获取当前内存中的值,称为 当前内存值 V,如果 A==V 则说明变量从未被其他线程修改过,此时将会写入新值 B,如果 A!=V 则说明变量已经被其他线程修改过,当前线程应当什么也不做或者再循环几个周期直到更新成功。

CAS通过避免同步,也就是避免了线程的上下文来极大的提高了同步处理的能力,比如在多线程下使用AtomicLong的无锁累加要比,使用synchronized关键字处理性能高出很多。

CAS 缺点

(1)ABA问题

通过上面的解释,我们知道CAS的原理是读取两个时刻的值,然后比较是否一致再决定是否更新,如果不一致,那么就需要多循环几次直到更新成功,这里面有一个问题假如第一次读到的预期值是A,然后在这段时间间隔内A的值变化成B又变化成A,这个时候其实它的值已经被更新过了,但是如果只比较值是没法判断出来是否更新过的,虽然对于一些计数,求和的操作不影响结果,但这是有缺陷的。 解决方法:通过版本号来判断,在每次更新后对应的版本号加1,这样一来在比较值之后如果相等,可以再次比较版本号来判断是否真的更新过。

JDK 1.5 以后的 AtomicStampedReference 类就提供了此种能力,其中的 compareAndSet 方法就是 首先检查当前引用是否等于预期引用,并且当前标志是否等于预期标志,如果全部相等,则以原子方式将该引用和该标志的值设置为给定的更新值。

(2)自旋多次循环导致的效率问题

上面说过CAS在判断两次读取的值不一样的时候会放弃操作,但为了保证结果正确,通常都会继续尝试循环再次发起CAS操作,如果连续多次CAS都失败,那么就会消耗大量的cpu资源。

(3)仅仅保证单个共享变量的原子操作?

CAS 只对单个共享变量有效,当操作涉及跨多个共享变量时 CAS 无效;不过从 JDK 1.5开始提供了 AtomicReference 类来保证引用对象之间的原子性,你可以把多个变量放在一个对象里来进行 CAS 操作

CAS与Java并发工具包的关系

java.util.concurrent包里面的工具类基本全部都是使用了CAS+volatile的乐观锁机制,也就是说Java并发工具包里面的大多数类底层是构建在CAS+volatile这种lock-free的机制上,所以拥有更好更高的并发效率。

总结

CAS是一项基于乐观锁的非阻塞技术(不需要线程的上下文切换),虽然CAS也有自己的一些缺点,但其通过操作系统底层提供的原子指令来实现lock-free的取得同步效果,在大多数情况下其效率和性能都要比synchronized和lock更好。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2018-07-17,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 我是攻城师 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 什么是CAS
  • CAS的原理
  • CAS 缺点
  • CAS与Java并发工具包的关系
  • 总结
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档