CAS(乐观锁)与ABA问题
cas是什么
CAS 全称 compare and swap 或者compare and exchange 比较并且交换。用于在没有锁的情况下,多个线程对同一个值的更新。

cas原理
例如,我们对一个int i进行递增操作。原来,为了线程安全,需要在递增代码上加一把锁 sync 。CAS的实现方式为:不去加锁,读取当前的值,将原值存入E中,然后计算,得到计算结果1,这个时候的新值存入V中。然后调用一个比较并交换的方法(底层汇编指令) 这个方法的作用是:如果E和现在的i值相等,说明没有地方对原值进行过操作。那么就将i的值替换为V的值。
如果有别的线程对i进行操作了(例如:i被修改为了2),那么将i的新值读取,然后再进行一次计算操作(将i=2 作为最初值,进行计算,得到结果3 )。然后再次调用比较并交换的方法,去进行值的更新操作。一直循环,直到更新成功(有的锁在更新到一定次数后,会进行锁升级,但是这个跟CAS没关系)
CAS底层原理
cas追代码的时候,可以追到native部分,但是,这里面提供的都是一些接口,但是没有具体的底层实现。具体的实现,需要看虚拟机的实现。
jvm的底层实现分为不同的公司的不同实现:Oracle的为hotSpot实现,这个是最常见的。我们可以看到代码的开源版本是openjdk。比如阿里的taobaoVM 。Ibm 的J9等。
在openjdk 中,unsafe类在实现CompareAndSwapInt 方法时,是调用的Atomic类的cmpxchg方法,然后一通方法调用,最终调用到汇编语言的 lock_if_mp(lock if multi processors 如果是多个cpu 那么会锁定) cmpxchg 这两个 指令 ,最终执行 lock cmpxchg 指令
这里的lock是一一个锁定北桥芯片的锁,用于解决当前cpu在进行比较和修改的时候的其他cpu对该值进行操作的问题。保证线程安全。
aba问题
在线程A进来时,读取到的值为1。在A进行计算的时候线程B取到了i的值,然后更新为了2。然后线程C又取到i=2的值,进行C的计算后,然后得到结果为1 又将i的值修改为了1 这个时候,线程A进行判断的时候,会认为在其计算过程中没有其他线程对i进行过操作。ABA有的会产生影响,有的不会,如果没有影响,不需要考虑该问题。
aba问题解决
解决方法:加上版本号,每个线程读取完,赋值的时候,修改版本号,比较的时候,对比i值+版本号。
CAS(乐观锁)与ABA问题的更多相关文章
- CAS(乐观锁)以及ABA问题
https://blog.csdn.net/wwd0501/article/details/88663621独占锁是一种悲观锁,synchronized就是一种独占锁:它假设最坏的情况,并且只有在确保 ...
- Java性能 -- CAS乐观锁
synchronized / Lock / CAS synchronized和Lock实现的同步锁机制,都属于悲观锁,而CAS属于乐观锁 悲观锁在高并发的场景下,激烈的锁竞争会造成线程阻塞,而大量阻塞 ...
- [数据库锁机制] 深入理解乐观锁、悲观锁以及CAS乐观锁的实现机制原理分析
前言: 在并发访问情况下,可能会出现脏读.不可重复读和幻读等读现象,为了应对这些问题,主流数据库都提供了锁机制,并引入了事务隔离级别的概念.数据库管理系统(DBMS)中的并发控制的任务是确保在多个事务 ...
- 并发之ATOMIC原子操作--CAS乐观锁原理(二)
1.乐观锁介绍 程序完成并发操作时,访问数据时每次不加锁,假设没有冲突去完成某项操作,如果因为冲突失败就重试,直到成功为止.就是当去做某个修改或其他操作的时候它认为不会有其他线程来做同样的操作(竞争) ...
- redis的高级事务CAS(乐观锁)
Optimistic locking using check-and-set(乐观锁) 乐观锁介绍:watch指令在redis事物中提供了CAS的行为.为了检测被watch的keys在是否有多个cli ...
- Java:CAS(乐观锁)
本文讲解CAS机制,主要是因为最近准备面试题,发现这个问题在面试中出现的频率非常的高,因此把自己学习过程中的一些理解记录下来,希望能对大家也有帮助. 什么是悲观锁.乐观锁?在java语言里,总有一些名 ...
- JAVA多线程学习四 - CAS(乐观锁)
本文讲解CAS机制,主要是因为最近准备面试题,发现这个问题在面试中出现的频率非常的高,因此把自己学习过程中的一些理解记录下来,希望能对大家也有帮助. 什么是悲观锁.乐观锁?在java语言里,总有一些名 ...
- java 乐观锁CAS
乐观锁是一种思想,本身代码里并没有lock或synchronized关键字进行修饰.而是采用一种version. 即先从数据库中查询一条记录得到version值,在更新这条记录时在where条件中对这 ...
- Java并发:乐观锁
作者:汤圆 个人博客:javalover.cc 简介 悲观锁和乐观锁都属于比较抽象的概念: 我们可以用拟人的手法来想象一下: 悲观锁:像有些人,凡事都往坏的想,做最坏的打算:在java中就表现为,总是 ...
随机推荐
- Linux——vim操作
查看文件:vim 文件名 进入vim命令后: Shift+g:到达文件底部 /搜索内容:搜索文件中字符串:点击“N”键,查看下一个搜索内容位置
- 有点干货 | Jdk1.8新特性实战篇(41个案例)
作者:小傅哥 博客:https://bugstack.cn - 汇总系列原创专题文章 沉淀.分享.成长,让自己和他人都能有所收获!
- RxJS 中的创建操作符
RxJs 中创建操作符是创建数据流的起点,这些操作符可以凭空创建一个流或者是根据其它数据形式创建一个流. Observable的构造函数可以直接创建一个数据流,比如: const $source=ne ...
- 基于 abp vNext 和 .NET Core 开发博客项目 - 定时任务最佳实战(三)
上一篇(https://www.cnblogs.com/meowv/p/12974439.html)完成了全网各大平台的热点新闻数据的抓取,本篇继续围绕抓取完成后的操作做一个提醒.当每次抓取完数据后, ...
- Bitwarden_rs搭建
最近LastPass网络极其不稳定,正好闲下来找到了Bitwarden_rs这个替代品,感觉不错,分享记录下部署过程. 一.Docker方式部署 #获取镜像 docker pull bitwarden ...
- jchdl - GSL实例 - Mux4(使用Mux)
https://mp.weixin.qq.com/s/GrYJ4KXEFRoLLmLnAGoMSA 原理图 参考链接 https://github.com/wjcdx/jchdl/blob/ma ...
- Linux磁盘与文件系统管理概要
Linux磁盘与文件系统管理 硬盘组成与分区 硬盘组成 圆形的盘片(主要记录数据) 机械手臂与磁头(可读取盘片上的数据) 主轴马达,转动盘片,让机械手臂的磁头在盘片上读取数据 扇区(Sector)为最 ...
- Java中那些烦人的位运算(&,|...)
& 和 && 相同点: 都表示"与"操作.这里的"与"和数学中的"与或非"中的"与"意义相同,都 ...
- Java实现 LeetCode 76 最小覆盖子串
76. 最小覆盖子串 给你一个字符串 S.一个字符串 T,请在字符串 S 里面找出:包含 T 所有字母的最小子串. 示例: 输入: S = "ADOBECODEBANC", T = ...
- Java实现 LeetCode 46 全排列
46. 全排列 给定一个没有重复数字的序列,返回其所有可能的全排列. 示例: 输入: [1,2,3] 输出: [ [1,2,3], [1,3,2], [2,1,3], [2,3,1], [3,1,2] ...