聊聊CAS - 面试官最喜欢问的并发编程专题
什么是CAS
学习Java并发编程,CAS(Compare And Set)机制都是一个不得不掌握的知识点。除了通过synchronized进行并发控制外,还可以通过CAS的方式控制,大家熟悉的ReentrantLock内部实现大量采用CAS进行控制。
CAS即Compare and Swap,即比较并交换。
CAS有三个操作数:内存值V、预期值A、要修改的值B,当且仅当预期值A和内存值V相同时,才将内存值修改为B,否则什么都不做。最后返回现在的V值。
CAS在任何一种情况下,都会返回V的真实值。(这个变量称为compare-and-set,无论操作是否成功都会返回。)CAS的意思是:“我认为V的值应该是A,如果是,那么将其赋值为B,若不是,则不修改,并告诉我应该为多少”。CAS是一项乐观技术,它抱着成功的希望进行更新,并且如果另一个线程在上次检查后更新了该变量,它能够发现错误。
CAS是由CPU指令来保证的,大多数的处理器都支持都实现了CAS指令。
CAS的优点
保证原子性的方式只有加锁,无论是内置锁(synchronized),还是显示锁(ReentrantLock),加锁意味着线程阻塞,如果竞争激烈,很可能导致频繁的线程上下文切换,从而大大降低了性能。
CAS机制是一种乐观锁,可以避免这些问题,使得执行效率很大的提高。
CAS典型应用场景
1、乐观锁思想
CAS解决并发的思路可以在很多场景中借鉴,在数据库使用中,可以采用这种方式来避免加锁操作,比如大家学习Hibernate时候的乐观锁。
2、Java Concurren
Java Concurrent包下大量使用CAS实现并发控制,比如Atomic*,ReentrantLock等。
比如AtomicInteger中,就使用CAS。
Java无法直接访问底层操作系统,而是通过本地(native)方法来访问。JVM开了一个后门,JDK中有一个类Unsafe,它提供了硬件级别的原子操作。
例如AtomicInteger的getAndSet方法:
public final int getAndSet(int newValue) {
     return unsafe.getAndSetInt(this, valueOffset, newValue);
 }unsafe类的getAndSetInt
public final int getAndSetInt(Object var1, long var2, int var4) {
    int var5;
    do {
        var5 = this.getIntVolatile(var1, var2);//根据地址和偏移量获取数值
    } while(!this.compareAndSwapInt(var1, var2, var5, var4));//重复尝试CAS操作
    return var5;
}通过调用unsafe的getlntVolatile(var1,var2),这是个native方法,其实就是获取var1中,var2偏移量处的值。var1就是Atomiclnteger对象本身,var2就是我们前面提到的偏移地址,这样我们就从内存里获取到现在偏移地址处的值了。
再通过compareAndSwapInt操作输入原值var5和目标值var4,如果操作成功返回,操作失败重新读取内存数据再次进行compareAndSwapInt直到成功。
CAS的缺点
CAS虽然很高效的解决原子操作,但是CAS仍然存在三大问题。ABA问题,循环时间长开销大和只能保证一个共享变量的原子操作。
1)ABA问题
CAS操作需要检查当前内存的变量值是否和刚刚读取的值相同,假设一个变量是A,变成了B,之后又变成了A,所以在进行CAS操作的时候检查到它的值没有发生变化。ABA的解决方案就是加上版本号1A->2B->3A,类似这种。java先行者们也给我们提供了AtomicStampedReference来解决ABA的问题,其中的预期标志也类似于版本号的功能。
2)循环时间长开销大
多线程竞争激烈的情况下进行CAS操作,会导致某些线程长时间空循环,也就是说它什么都没做,只是不停地在浪费处理器的处理时间而已。
3)只能保证一个共享变量的原子操作
一个共享变量的操作可以用CAS保证其原子性,多个共享变量的操作,循环CAS是无法保证其原子性的。有个取巧的办法,java先行者设计的AtomicReference类,我们可以通过将多个变量放到一个对象里面,然后由AtomicReference进行原子性地更新。
欢迎大家讨论并发的相关问题,也可以加我的头条号: IT技术研习社 。

聊聊CAS - 面试官最喜欢问的并发编程专题的更多相关文章
- 基础面试,为什么面试官总喜欢问String?
		关于 Java String,这是面试的基础,但是还有很多童鞋不能说清楚,所以本文将简单而又透彻的说明一下那个让你迷惑的 String 在 Java 中,我们有两种方式创建一个字符串 String x ... 
- 一线大厂面试官最喜欢问的15道Java多线程面试题
		前言 在任何Java面试当中多线程和并发方面的问题都是必不可少的一部分.如果你想获得更多职位,那么你应该准备很多关于多线程的问题. 他们会问面试者很多令人混淆的Java线程问题.面试官只是想确信面试者 ... 
- 2019年面试官最喜欢问的28道ZooKeeper面试题
		前言 ZooKeeper 是一个分布式的,开放源码的分布式应用程序协调服务.它是一个为分布式应用提供一致性服务的软件,提供的功能包括:配置维护.域名服务.分布式同步.组服务等. ZooKeeper 的 ... 
- 阿里面试官最喜欢问的21个HashMap面试题
		1.HashMap 的数据结构? A:哈希表结构(链表散列:数组+链表)实现,结合数组和链表的优点.当链表长度超过 8 时,链表转换为红黑树. transient Node<K,V>\[\ ... 
- 深度分析:面试腾讯,阿里面试官都喜欢问的String源码,看完你学会了吗?
		前言 最近花了两天时间,整理了一下String的源码.这个整理并不全面但是也涵盖了大部分Spring源码中的方法.后续如果有时间还会将剩余的未整理的方法更新到这篇文章中.方便以后的复习和面试使用.如果 ... 
- 2021超详细的HashMap原理分析,面试官就喜欢问这个!
		一.散列表结构 散列表结构就是数组+链表的结构 二.什么是哈希? Hash也称散列.哈希,对应的英文单词Hash,基本原理就是把任意长度的输入,通过Hash算法变成固定长度的输出 这个映射的规则就是对 ... 
- 走向DBA[MSSQL篇] 面试官最喜欢的问题 ----索引+C#面试题客串
		原文:走向DBA[MSSQL篇] 面试官最喜欢的问题 ----索引+C#面试题客串 对大量数据进行查询时,可以应用到索引技术.索引是一种特殊类型的数据库对象,它保存着数据表中一列或者多列的排序结果,有 ... 
- Java面试官最常问的volatile关键字
		在Java相关的职位面试中,很多Java面试官都喜欢考察应聘者对Java并发的了解程度,以volatile关键字为切入点,往往会问到底,Java内存模型(JMM)和Java并发编程的一些特点都会被牵扯 ... 
- 大厂面试官最常问的@Configuration+@Bean(JDKConfig编程方式)
		大厂面试官最常问的@Configuration+@Bean(JDKConfig编程方式) 现在大部分的Spring项目都采用了基于注解的配置,采用了@Configuration 替换标签的做法.一 ... 
随机推荐
- Leetcode13_罗马数字转整数
			题目 罗马数字包含以下七种字符: I, V, X, L,C,D 和 M. 字符 数值I 1V 5X 10L 50C 100D 500M 1000例如, 罗马数字 2 写做 II ,即为两个并列的 1. ... 
- 59)PHP,管理员表中所存在的项
			用户ID 用户名 用户密码 用户权限(就是他的角色等级,比如是1级 2级, 三级等等) 上次登录的IP 上次登录的时间 
- http接口与webservice接口的区别
			常见的API接口有两类:http接口和webservice接口. http接口走http协议,通过路径来区分调用方法,请求报文一般是key-value形式的,返回报文一般是json串,常用的是get和 ... 
- Linux正则表达式及扩展的正则表达式
			基本正则表达式: 扩展的正则表达式: 
- webpack中使用ECharts
			npm安装ECharts 引入ECharts 通过 npm 上安装的 ECharts 和 zrender 会放在node_modules目录下.可以直接在项目代码中 require('echarts' ... 
- [洛谷P4720] [模板] 扩展卢卡斯
			题目传送门 求组合数的时候,如果模数p是质数,可以用卢卡斯定理解决. 但是卢卡斯定理仅仅适用于p是质数的情况. 当p不是质数的时候,我们就需要用扩展卢卡斯求解. 实际上,扩展卢卡斯=快速幂+快速乘+e ... 
- 吴裕雄--天生自然 R语言开发学习:聚类分析(续一)
			#-------------------------------------------------------# # R in Action (2nd ed): Chapter 16 # # Clu ... 
- 吴裕雄--天生自然 R语言开发学习:重抽样与自助法(续一)
			#-------------------------------------------------------------------------# # R in Action (2nd ed): ... 
- Python测试开发-浅谈如何自动化生成测试脚本
			Python测试开发-浅谈如何自动化生成测试脚本 原创: fin 测试开发社区 前天 阅读文本大概需要 6.66 分钟. 一 .接口列表展示,并选择 在右边,点击选择要关联的接口,区分是否要登录, ... 
- nginx增加访问验证
			使用OpenSSL实用程序创建密码文件 如果您的服务器上安装了OpenSSL,则可以创建没有附加软件包的密码文件.我们将在/ etc / nginx配置目录中创建一个名为.htpasswd的隐藏文件来 ... 
