您现在的位置是:亿华云 > 系统运维
并发扣款一致性优化,CAS下ABA问题,这个话题还没聊完!!!
亿华云2025-10-09 03:34:00【系统运维】5人已围观
简介上一篇答星球水友提问,《并发扣款,如何保证数据的一致性?》中提到:用CAS乐观锁,可以在尽量不影响吞吐量的情况下,保证数据的一致性。大家有非常多的留言,大概有这么几类:是否存在ABA问题?
上一篇答星球水友提问,扣款《并发扣款,致性如何保证数据的优化一致性?》中提到:用CAS乐观锁,可以在尽量不影响吞吐量的话题还没情况下,保证数据的聊完一致性。
大家有非常多的扣款留言,大概有这么几类:
是致性否存在ABA问题? 为什么不能用: UPDATE t_yue SET moneymoney=money-$diff AND money>=$diff; 能否借助redis事务来扣减余额;画外音:请务必阅读前序文章:《并发扣款,如何保证数据的优化一致性?》。
问题比较多,话题还没今天先聊第一个问题,聊完ABA。扣款
什么是致性ABA问题?
CAS乐观锁机制确实能够提升吞吐,并保证一致性,优化但在极端情况下可能会出现ABA问题。话题还没
考虑如下操作:
并发1(上):获取出数据的聊完初始值是A,后续计划实施CAS乐观锁,期望数据仍是A的时候,修改才能成功 并发2:将数据修改成B 并发3:将数据修改回A 并发1(下):CAS乐观锁,检测发现初始值还是A,进行数据修改上述并发环境下,并发1在修改数据时,虽然还是高防服务器A,但已经不是初始条件的A了,中间发生了A变B,B又变A的变化,此A已经非彼A,数据却成功修改,可能导致错误,这就是CAS引发的所谓的ABA问题。
余额操作,出现ABA问题并不会对业务产生影响,因为对于“余额”属性来说,前一个A为100余额,与后一个A为100余额,本质是相同的。
但其他场景未必是这样,举一个堆栈操作的例子:
并发1(上):读取栈顶的元素为“A1”
并发2:进行了2次出栈
并发3:又进行了1次出栈
并发1(下):实施CAS乐观锁,发现栈顶还是“A1”,于是修改为A2
此时会出现系统错误,因为此“A1”非彼“A1”
ABA问题可以怎么优化?
ABA问题导致的原因,站群服务器是CAS过程中只简单进行了“值”的校验,再有些情况下,“值”相同不会引入错误的业务逻辑(例如余额),有些情况下,“值”虽然相同,却已经不是原来的数据了(例如堆栈)。
因此,CAS不能只比对“值”,还必须确保是原来的数据,才能修改成功。
常见的实践是,将“值”比对,升级为“版本号”的比对,一个数据一个版本,版本变化,即使值相同,也不应该修改成功。
余额并发读写例子,引入版本号的具体实践如下:
(1)余额表要升级。
t_yue(uid, money)升级为:
t_yue(uid, money, version)(2)查询余额时,同时查询版本号。
SELECT money FROM t_yue WHERE sid=$sid升级为:
SELECT money,version FROM t_yue WHERE sid=$sid假设有并发操作,b2b供应网都会将版本号查询出来。
(3)设置余额时,必须版本号相同,并且版本号要修改。旧版本“值”比对:
UPDATE t_yue SET money=38 WHERE uid=$uid AND money=100升级为“版本号”比对:
UPDATE t_yue SET money=38, version=$version_new WHERE uid=$uid AND version=$version_old此时假设有并发操作,首先操作的请求会修改版本号,并发操作会执行失败。
画外音:version通用,本例是强行用version举例而已,实际上本例可以用余额“值”比对。
总结
select&set业务场景,在并发时会出现一致性问题 基于“值”的CAS乐观锁,可能导致ABA问题 CAS乐观锁,必须保证修改时的“此数据”就是“彼数据”,应该由“值”比对,优化为“版本号”比对思路比结论重要。
【本文为专栏作者“58沈剑”原创稿件,转载请联系原作者】
戳这里,看该作者更多好文
很赞哦!(5)
相关文章
- 5、使用企业名称的英文名称作为域名也是国内许多企业选择域名的一种方式,特别适合一些与计算机、网络和通信相关的行业。
- 换域名后会影响网站功能吗?新手该注意什么?
- 微软推出 VS Code Server,用平板也能远程开发!
- 从零开始写项目第一篇【搭建环境】
- 3、不明先知,根据相关征兆预测可能发生的事件,以便提前做好准备,赶紧注册相关域名。;不差钱域名;buchaqian抢先注册,就是这种敏感类型。预言是最敏感的状态。其次,你应该有眼力。所谓眼力,就是善于从社会上时不时出现的各种热点事件中获取与事件相关的域名资源。眼力的前提是对域名领域的熟悉和丰富的知识。
- Spring Cloud:使用Ribbon实现负载均衡详解(上)
- 未实名的域名能用吗?新手需要了解哪些?
- Eureka中读写锁的奇思妙想,太顶了
- 打开https://www.aizhan.com/输入自己想要查询的域名然后按回车键,如果做过网站都会有数据显示出来
- 扩展我们的分析处理服务(Smartly.io):使用 Citus 对 PostgreSQL 数据库进行分片
站长推荐
用户邮箱的静态密码可能已被钓鱼和同一密码泄露。在没有收到安全警报的情况下,用户在适当的时间内不能更改密码。在此期间,攻击者可以随意输入帐户。启用辅助身份验证后,如果攻击者无法获取移动电话动态密码,他将无法进行身份验证。这样,除非用户的电子邮件密码和手机同时被盗,否则攻击者很难破解用户的邮箱。
怎么购买美国域名?使用美国域名有什么优势?
单元测试时静态方法注意点
JVM的特性,通过代码来揭秘运行时数据区
比较短的域名方便用户记忆和传播,它带来的好处往往会超过其他类型的域名,如果你非要域名短而且还要包含关键词,那么往往会事与愿违,现在这种域名基本上是可遇而不可求的。
HTTP/3正式发布,深入理解HTTP/3协议
JVM解析
分布式 PostgreSQL 集群(Citus),官方快速入门教程