业务场景
一般的商城业务中,我们经常会遇到一种场景,某商品库存只剩最后一件,但此时有多个用户同时下单,如果没有做任何的处理,多个用户下单时进程读取到的库存为1,程序判断符合扣库存条件,下单成功,就会造成库存变成负数的情况
解决方案
思路
乐观锁通过增加一个每次更新库存都会自增的version字段来解决这个问题,每个进程读取库存字段的同时,也会读取version的值,在更新库存同时更新version,并增加对version字段值的判断
代码实现
<!-- 乐观锁更新库存 -->
<update id="newMinStock" parameterType="java.util.Map">
update goods
set stock= stock - 1,version = #{version} + 1
where id = #{id} and version = #{version}
</update>
假设有两个进程都读取到了stock的值为1,version默认值为0,则这两个进程执行的更新语句都是update goods set stock=stock-1,version=version+1 where id=1 and version=0,当其中一个进程执行成功之后,数据库中version的值就会变为1,剩下的进程不会执行成功,这样保证了stock的值不会小于0
Map<String, Object> map = new HashMap<>();
map.put("id", id);
map.put("version", version);
int result = goodsMapper.newMinStock(map);
if (result != 1) {
throw new RuntimeException("更新商品库存失败");
}
在业务层判断更新库存是否成功,进行相应的业务处理