浅谈CAS以及CAS在java中应用

浅谈CAS以及CAS在java中应用

cas是什么?

cas是compareandswap的简称,从字面上理解就是比较并更新,简单来说:从某一内存上取值V,和预期值A进行比较,如果内存值V和预期值A的结果相等,那么我们就把新值B更新到内存,如果不相等,那么就重复上述操作直到成功为止。

cas能做什么?

上面我们了解了cas是什么了,那么它能解决什么问题呢?它可以解决多线程并发安全的问题,以前我们对一些多线程操作的代码都是使用synchronize关键字,来保证线程安全的问题;现在我们将cas放入到多线程环境里我们看一下它是怎么解决的,我们假设有A、B两个线程同时执行一个int值value自增的代码,并且同时获取了当前的value,我们还要假设线程B比A快了那么0.00000001s,所以B先执行,线程B执行了cas操作之后,发现当前值和预期值相符,就执行了自增操作,此时这个value = value + 1;然后A开始执行,A也执行了cas操作,但是此时value的值和它当时取到的值已经不一样了,所以此次操作失败,重新取值然后比较成功,然后将value值更新,这样两个线程进入,value值自增了两次,符合我们的预期。

cas在java中的应用

是不是感觉cas很好用,那么在java中有对应的实现吗?有的!java从jdk1.5就将cas引入并且使用了,java中的Atomic系列就是使用cas实现的,下面我们就用AtomicInteger类看一下java是怎么实现的吧。

进入到AtomicInteger类里边之后,我们发现它使用volatile声明了一个变量,至于volatile有什么特性,我就不详细赘述了,简单来说volatile声明这个变量是易变的,当线程拿到这个值并且更新之后还要将更新后的值同步到主内存里边,供之后的线程调用。

好了,了解了volatile的特性之后,我们再来看一下它怎么实现自增的吧。
在这里插入图片描述

AtomicInteger有一个incrementAndGet的自增方法,在一个循环里,每次去获取当前的值current,然后将当前值current+1赋值给next,然后将current和next放到compareAndSet中进行比较,如果返回true那么就return next的值,如果失败,那么继续进行上述操作,是不是很眼熟这个操作,是的,你没看错,这里就是使用了cas操作,看到这里是不是感觉java很直白,哈哈。

好的,我们再来看compareAndSet是不是如我们所想的那样使用了cas呢,我们再进入compareAndSet方法中一探究竟。

在这里插入图片描述
可以看到这个compareAndSet方法有两个参数,分别叫expect和update,从字面上理解就是预期的值和更新的值,OK,我们再来看里边,里边调用了一个compareAndSwapInt的方法,有四个参数分别是当前的值this、valueOffset、预期值expect、更新的值update,其中expect和update是通过参数传过来的,你们还记得这两个值分别是什么吗?不记得的童鞋们请网上翻,没错就是incrementAndGet方法中的current和next,好的,我们来梳理一下this当前值和预期值expect也就是current进行比较,如果相等,就把值更新为update也就是next,这样此次自增操作完成!至于valueOffset容我买个关子。

CAS有没有什么不好的隐患呢?

毫无疑问肯定有的!
1、首先就是经典的ABA问题
何为ABA呢?我们还是以两个线程L、N进行自增操作为例,线程L、N同时获取当前的值A,只不过此时线程N比较快,它在L操作之前,进行了两次操作,第一次将值从A 改为了B,之后又将B改为了A,那么在线程L操作的 时候发现当前的值还是A,符合预期,那么它也会更新成功,从操作上看并没有什么不对,更新成功也是对的,但是这样是有隐患的,这个网上有好多关于ABA问题隐患的解读,我觉得有一个老哥使用链表的表述最为贴切,这个是我很久之前看的,我现在也找不到这个老哥关于这个问题解读的帖子了,大家自行搜索一下吧,为了解决这个问题,java引入了版本的概念,相当于上述操作变为了A1----B2----A3,这样就非常明确了,这个版本相信大家也猜到那就是valueOffset,所以在AtomicInteger中进行cas操作时除了this、expect、update之外还有一个valueOffset的参数进行版本的区分,就是为了解决ABA问题的。

2、长时间自旋非常消耗资源
先说一下什么叫自旋,自旋就是cas的一个操作周期,如果一个线程特别倒霉,每次获取的值都被其他线程的修改了,那么它就会一直进行自旋比较,直到成功为止,在这个过程中cpu的开销十分的大,所以要尽量避免。

第一次写博客,如有问题,请大家多多指正,thanks!
联系方式:
邮件:lnjavamail@163.com
微信:ln_coder

  • 79
    点赞
  • 165
    收藏
    觉得还不错? 一键收藏
  • 26
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 26
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值