场景:减库存
知识点:
setIfAbsent():设置值之前判断key是否存在,setIfAbsent(key,vlue,时长,时间类型)
* setIfAbsent()是redis(setnx)在java中的用法
思路:
1.生成一个库存锁,把存库数量stock和库存的锁result同时放在redis中,更改存库数量时,先判断库存锁是否有效和存在。
2.如果库存锁存在返回一个错误提示
3.如果存库锁不存在,拿取库存数量执行具体代码逻辑
4.执行完毕后删除锁
5.整个业务逻辑用try包裹,删除锁放在finally里,保证都能释放锁
5.后续其他线程同样这样执行。
锁的设计:
1.clientId:标识本次方法执行的身份id
2.利用setIfAbsent()的特性,设置锁的key:lockKey,设置锁的value:clientId,设置锁的失效时间:30,设置时间类型:秒
具体代码实现:
缺陷:
身份验证和删除锁不是原子操作,如果在拿到锁并且身份验证通过后还没有删除锁时,jvm在锁失效的瞬间发生了gc(程序停顿了一瞬间)。一样会导致删除了别的线程的锁,出现多扣库存的情况
针对这种情况解决办法:可以启用一个监听线程,对锁进行续命(后续更新补充)