Java编程cas操作全面解析
发布时间 - 2026-01-11 03:15:07 点击率:次CAS 指的是现代 CPU 广泛支持的一种对内存中的共享数据进行操作的一种特殊指令。这个指令会对内存中的共享数据做原子的读写操作。

简单介绍一下这个指令的操作过程:首先,CPU 会将内存中将要被更改的数据与期望的值做比较。然后,当这两个值相等时,CPU 才会将内存中的数值替换为新的值。否则便不做操作。最后,CPU 会将旧的数值返回。这一系列的操作是原子的。它们虽然看似复杂,但却是 Java 5 并发机制优于原有锁机制的根本。简单来说,CAS 的含义是“我认为原有的值应该是什么,如果是,则将原有的值更新为新值,否则不做修改,并告诉我原来的值是多少”。(这段描述引自《Java并发编程实践》)
简单的来说,CAS有3个操作数,内存值V,旧的预期值A,要修改的新值B。当且仅当预期值A和内存值V相同时,将内存值V修改为B,否则返回V。这是一种乐观锁的思路,它相信在它修改之前,没有其它线程去修改它;而Synchronized是一种悲观锁,它认为在它修改之前,一定会有其它线程去修改它,悲观锁效率很低。
下面看一个简单的例子:
if(a==b) {
a++;
}
试想一下如果在做a++之前a的值被改变了怎么办?a++还执行吗?出现该问题的原因是在多线程环境下,a的值处于一种不定的状态。采用锁可以解决此类问题,但CAS也可以解决,而且可以不加锁。
int expect = a;
if(a.compareAndSet(expect,a+1)) {
doSomeThing1();
} else {
doSomeThing2();
}
这样如果a的值被改变了a++就不会被执行。
按照上面的写法,a!=expect之后,a++就不会被执行,如果我们还是想执行a++操作怎么办,没关系,可以采用while循环
while(true) {
int expect = a;
if (a.compareAndSet(expect, a + 1)) {
doSomeThing1();
return;
} else {
doSomeThing2();
}
}
采用上面的写法,在没有锁的情况下实现了a++操作,这实际上是一种非阻塞算法。
应用
java.util.concurrent.atomic包中几乎大部分类都采用了CAS操作,以AtomicInteger为例,看看它几个主要方法的实现:
public final int getAndSet(int newValue) {
for (;;) {
int current = get();
if (compareAndSet(current, newValue))
return current;
}
}
getAndSet方法JDK文档中的解释是:以原子方式设置为给定值,并返回旧值。原子方式体现在何处,就体现在compareAndSet上,看看compareAndSet是如何实现的:
public final boolean compareAndSet(int expect, int update) {
return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
}
不出所料,它就是采用的Unsafe类的CAS操作完成的。
再来看看a++操作是如何实现的:
public final int getAndIncrement() {
for (;;) {
int current = get();
int next = current + 1;
if (compareAndSet(current, next))
return current;
}
}
几乎和最开始的实例一模一样,也是采用CAS操作来实现自增操作的。
++a操作和a++操作类似,只不过返回结果不同罢了
public final int incrementAndGet() {
for (;;) {
int current = get();
int next = current + 1;
if (compareAndSet(current, next))
return next;
}
}
此外,java.util.concurrent.ConcurrentLinkedQueue类全是采用的非阻塞算法,里面没有使用任何锁,全是基于CAS操作实现的。CAS操作可以说是JAVA并发框架的基础,整个框架的设计都是基于CAS操作的。
缺点:
1、ABA问题
维基百科上给了一个活生生的例子——
你拿着一个装满钱的手提箱在飞机场,此时过来了一个火辣性感的|美女|,然后她很暖昧地挑逗着你,并趁你不注意的时候,把用一个一模一样的手提箱和你那装满钱的箱子调了个包,然后就离开了,你看到你的手提箱还在那,于是就提着手提箱去赶飞机去了。
这就是ABA的问题。
CAS操作容易导致ABA问题,也就是在做a++之间,a可能被多个线程修改过了,只不过回到了最初的值,这时CAS会认为a的值没有变。a在外面逛了一圈回来,你能保证它没有做任何坏事,不能!!也许它讨闲,把b的值减了一下,把c的值加了一下等等,更有甚者如果a是一个对象,这个对象有可能是新创建出来的,a是一个引用呢情况又如何,所以这里面还是存在着很多问题的,解决ABA问题的方法有很多,可以考虑增加一个修改计数,只有修改计数不变的且a值不变的情况下才做a++,也可以考虑引入版本号,当版本号相同时才做a++操作等,这和事务原子性处理有点类似!
2、比较花费CPU资源,即使没有任何争用也会做一些无用功。
3、会增加程序测试的复杂度,稍不注意就会出现问题。
总结
可以用CAS在无锁的情况下实现原子操作,但要明确应用场合,非常简单的操作且又不想引入锁可以考虑使用CAS操作,当想要非阻塞地完成某一操作也可以考虑CAS。不推荐在复杂操作中引入CAS,会使程序可读性变差,且难以测试,同时会出现ABA问题。
以上是本文关于Java编程cas操作的全部内容,希望对大家能有所帮助。
# java
# cas操作
# cas编程
# Java多线程CAS操作原理代码实例解析
# Java Unsafe类实现原理及测试代码
# java Unsafe详细解析
# Java并发编程学习之Unsafe类与LockSupport类源码详析
# 一篇看懂Java中的Unsafe类
# Java中的魔法类:sun.misc.Unsafe示例详解
# 简单谈一谈Java中的Unsafe类
# Java中unsafe操作实例总结
# Java CAS操作与Unsafe类详解
# 是一个
# 会将
# 是在
# 是一种
# 情况下
# 不做
# 如何实现
# 都是
# 只不过
# 几个
# 这一
# 就会
# 改变了
# 去了
# 告诉我
# 过了
# 却是
# 也就
# 多个
# 有可能
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
使用spring连接及操作mongodb3.0实例
Win11怎样安装网易有道词典_Win11安装词典教程【步骤】
如何快速查询网站的真实建站时间?
Google浏览器为什么这么卡 Google浏览器提速优化设置步骤【方法】
如何快速搭建安全的FTP站点?
Laravel如何使用Gate和Policy进行授权?(权限控制)
Laravel如何使用集合(Collections)进行数据处理_Laravel Collection常用方法与技巧
怎样使用JSON进行数据交换_它有什么限制
Laravel如何实现邮箱地址验证功能_Laravel邮件验证流程与配置
如何用PHP工具快速搭建高效网站?
WEB开发之注册页面验证码倒计时代码的实现
深圳网站制作的公司有哪些,dido官方网站?
C语言设计一个闪闪的圣诞树
Python文件流缓冲机制_IO性能解析【教程】
Internet Explorer官网直接进入 IE浏览器在线体验版网址
成都网站制作公司哪家好,四川省职工服务网是做什么用?
Laravel如何实现多表关联模型定义_Laravel多对多关系及中间表数据存取【方法】
为什么要用作用域操作符_php中访问类常量与静态属性的优势【解答】
长沙企业网站制作哪家好,长沙水业集团官方网站?
BootStrap整体框架之基础布局组件
Laravel的辅助函数有哪些_Laravel常用Helpers函数提高开发效率
Laravel辅助函数有哪些_Laravel Helpers常用助手函数大全
如何在阿里云虚拟主机上快速搭建个人网站?
Windows10如何更改计算机工作组_Win10系统属性修改Workgroup
Laravel怎么返回JSON格式数据_Laravel API资源Response响应格式化【技巧】
详解vue.js组件化开发实践
如何在万网自助建站中设置域名及备案?
高防服务器:AI智能防御DDoS攻击与数据安全保障
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
如何为不同团队 ID 动态生成多个非值班状态按钮
深圳网站制作培训,深圳哪些招聘网站比较好?
专业商城网站制作公司有哪些,pi商城官网是哪个?
百度输入法ai面板怎么关 百度输入法ai面板隐藏技巧
宙斯浏览器视频悬浮窗怎么开启 边看视频边操作其他应用教程
Laravel Eloquent性能优化技巧_Laravel N+1查询问题解决
Laravel如何实现用户密码重置功能?(完整流程代码)
如何在浏览器中启用Flash_2025年继续使用Flash Player的方法【过时】
手机怎么制作网站教程步骤,手机怎么做自己的网页链接?
电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?
JavaScript如何实现继承_有哪些常用方法
HTML 中动态设置元素 name 属性的正确语法详解
太平洋网站制作公司,网络用语太平洋是什么意思?
Laravel如何实现密码重置功能_Laravel密码找回与重置流程
大连 网站制作,大连天途有线官网?
Laravel如何发送邮件和通知_Laravel邮件与通知系统发送步骤
高防服务器租用如何选择配置与防御等级?
JS碰撞运动实现方法详解
Claude怎样写约束型提示词_Claude约束提示词写法【教程】
如何用AI一键生成爆款短视频文案?小红书AI文案写作指令【教程】
Laravel如何使用Service Provider注册服务_Laravel服务提供者配置与加载

