上一篇答星球水友提问,扣款《并发扣款,致性如何保证数据的优化一致性?》中提到:用CAS乐观锁,可以在尽量不影响吞吐量的话题还没情况下,保证数据的聊完一致性。 大家有非常多的扣款留言,大概有这么几类: 画外音:请务必阅读前序文章:《并发扣款,如何保证数据的优化一致性?》。 问题比较多,话题还没今天先聊第一个问题,聊完ABA。扣款 什么是致性ABA问题? CAS乐观锁机制确实能够提升吞吐,并保证一致性,优化但在极端情况下可能会出现ABA问题。话题还没 考虑如下操作: 上述并发环境下,并发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)余额表要升级。 升级为: (2)查询余额时,同时查询版本号。 升级为: 假设有并发操作,网站模板都会将版本号查询出来。 (3)设置余额时,必须版本号相同,并且版本号要修改。旧版本“值”比对: 升级为“版本号”比对: 此时假设有并发操作,首先操作的请求会修改版本号,并发操作会执行失败。 画外音:version通用,本例是强行用version举例而已,实际上本例可以用余额“值”比对。 总结 思路比结论重要。 【本文为专栏作者“58沈剑”原创稿件,转载请联系原作者】 戳这里,看该作者更多好文