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中就表现为,总是 ...
随机推荐
- MySQL创建用户,并设置指定访问数据库
一.创建用户并授权 1. 登录mysql mysql -u root -q输入密码2. 创建数据库(已有数据库就不需要建立) create database newDB;//以创建newDB为例3. ...
- JS中Class的两种写法
写法一:使用函数: var ClassName = function() { this.message = 'dat.gui'; this.speed = 0.8; this.displayOutli ...
- [PHP自动化-进阶]005.Snoopy采集框架介绍
引言:Snoopy是一个php类,用来模仿web浏览器的功能,它能完成获取网页内容和发送表单的任务. **官方网址:http://snoopy.sourceforge.net/** 简单一句话表达:& ...
- 搭建SpringCloud微服务框架:一、结构和各个组件
搭建微服务框架(结构和各个组件) 简介 SQuid是基于Spring,SpringBoot,使用了SpringCloud下的组件进行构建,目的是想搭建一套可以快速开发部署,并且很好上手的一套微服务框架 ...
- 01 . Nginx简介及部署
Nginx简介 Nginx(发音同engine x)是一个异步框架的 Web 服务器,也可以用作反向代理,负载平衡器 和 HTTP 缓存.该软件由 Igor Sysoev 创建,并于2004年首次公开 ...
- 北京理工大学复试上机--2001A
1.编写程序,计算下列分段函数 y=f(x)的值. y = -x + 2.5, 0 <= x < 2 y = 2 - 1.5 (x - 3) (x - 3), 2 <= x < ...
- Java-接口(另类抽象)
1.1 特点 用interface定义 接口中所有成员变量都默认是由public static final修饰的 接口中所有方法都默认是由public abstract修饰的 接口没有构造器 接口采用 ...
- Rocket - devices - CanHaveBuiltInDevices
https://mp.weixin.qq.com/s/C9iktVr4hnQ8lM0CiWtedQ 简单介绍CanHaveBuiltInDevices的实现. 1. HasBuiltInDeviceP ...
- Java并发编程 (九) 线程调度-线程池
个人博客网:https://wushaopei.github.io/ (你想要这里多有) 声明:实际上,在开发中并不会普遍的使用Thread,因为它具有一些弊端,对并发性能的影响比较大,如下: ...
- Java实现 蓝桥杯VIP 算法训练 奇偶判断
问题描述 能被2整除的数称为偶数,不能被2整除的数称为奇数.给一个整数x,判断x是奇数还是偶数. 输入格式 输入包括一个整数x,0<=x<=100000000. 输出格式 如果x是奇数,则 ...