目录
什么是超卖?
乐观锁和悲观锁的定义
悲观锁:
乐观锁:
乐观锁的实现方式
1.版本号
2.CAS法
什么是超卖?
举个例子:订单系统中,用户在执行下单操作时,可能同一时间有无数个用户同时下单,当a用户的请求查询当前商品库存时,发现当前的商品剩余5件,在执行生成订单并减少库存时,线程切换了,此时b用户执行了查询操作,发现还是剩余5件,并进行了下单操作,这样就导致了这件商品被卖掉了两次
正常的情况如下:
发生线程切换,产生异常的情况:
乐观锁和悲观锁的定义
悲观锁:
认为线程安全问题一定会发生,因此在操作数据之前先获取锁,确保线程串行执行。例如Synchronized、Lock都属于悲观锁
乐观锁:
认为线程安全问题不一定会发生,因此不加锁,只是在更新数据时去判断有没有其它线程对数据做了修改。
如果没有修改则认为是安全的,自己才更新数据。
如果已经被其它线程修改说明发生了安全问题,此时可以重试或异常
乐观锁的实现方式
1.版本号
给商品加上版本号字段,如果查询到就让其version=1,在修改执行的时候,先判断版本号是不是正确的,如果是让其版本号发生变化,并执行扣减,如果不是就说明当前商品已经卖出
2.CAS法
CAS流程如下:
- 获取目标内存位置的当前值。
- 检查当前值是否与预期值相等。
- 如果相等,则将新值写入目标内存位置;否则,放弃写入操作,可能是重新读取当前值并重试整个CAS操作。
比如当前的订单系统中,就可以使用查询到的库存作为预期值,修改的时候进行判定,如果是库存和第一次查询到的一样就执行,不一样就取消执行,这样就能够保证原子性
具体实现只需要更改sql语句就可以做到
UPDATE users
SET stock=stock-1
WHERE id = 10 and stock = #{第一次查询到的库存};