多线程超卖问题的解决方案:悲观锁与乐观锁
创作时间:
作者:
@小白创作中心
多线程超卖问题的解决方案:悲观锁与乐观锁
引用
CSDN
1.
https://blog.csdn.net/qq_52351978/article/details/136413998
在多线程环境下,如何避免库存超卖问题?本文将从悲观锁和乐观锁两种解决方案入手,详细探讨这一问题的原理和实现方式。
1. 超卖问题
超卖问题通常出现在多线程环境中,当多个线程同时访问和修改共享资源时,可能会导致资源被过度消耗。具体来说,假设线程1查询库存发现库存大于1,正准备扣减库存时,线程2也查询到库存大于1并尝试扣减,最终导致库存被过度扣减。
2. 乐观锁解决方案
乐观锁的基本思想是认为线程冲突不会经常发生,因此在操作数据时不会立即加锁,而是假设操作能够成功。如果在提交数据时发现数据已被其他线程修改,则操作失败并重试。
解决原理
CAS(Compare and Swap)操作是乐观锁的一种实现方式。在多线程环境下,只有第一个执行CAS操作的线程能够成功修改库存信息,其他线程由于版本号不匹配而失败。这种机制可以有效避免多个线程同时修改库存信息的问题。
业务场景
在使用乐观锁时,如果多个线程同时获取到相同的库存值并尝试扣减,只有第一个线程能够成功,其他线程会因为版本号不匹配而失败。例如,如果有100个线程同时获取到库存为100,只有1个线程能成功扣减,其他线程都会失败。
弊端
使用乐观锁时,如果并发量很大,可能会导致很多操作失败,从而降低系统的整体性能。例如,即使库存有100个,但在高并发场景下,由于成功率低,最终库存可能仍然很高。
3. 悲观锁解决方案
悲观锁假设线程冲突经常发生,因此在操作数据前会先获取锁,确保线程串行执行。以下是使用悲观锁解决超卖问题的代码示例:
@Transactional
public Result createVoucherOrder(Long voucherId) {
Long userId = UserHolder.getUser().getId();
// 5.1.查询订单
int count = query().eq("user_id", userId).eq("voucher_id", voucherId).count();
// 5.2.判断是否存在
if (count > 0) {
// 用户已经购买过了
return Result.fail("用户已经购买过一次!");
}
// 6.扣减库存
boolean success = seckillVoucherService.update()
.setSql("stock = stock - 1") // set stock = stock - 1
.eq("voucher_id", voucherId).gt("stock", 0) // where id = ? and stock > 0
.update();
if (!success) {
// 扣减失败
return Result.fail("库存不足!");
}
// 7.创建订单
VoucherOrder voucherOrder = new VoucherOrder();
// 7.1.订单id
long orderId = redisIdWorker.nextId("order");
voucherOrder.setId(orderId);
// 7.2.用户id
voucherOrder.setUserId(userId);
// 7.3.代金券id
voucherOrder.setVoucherId(voucherId);
save(voucherOrder);
// 7.返回订单id
return Result.ok(orderId);
}
注意事项
- 锁的对象需要调用intern()
- 当使用字符串常量或调用
String类的intern()方法创建字符串时,会将字符串添加到字符串常量池中。而使用new String()创建字符串时,会在堆内存中创建一个新的字符串对象。 - 在多线程环境下,如果多个线程同时访问到相同的字符串,但创建了不同的字符串对象,会导致锁失效的问题。因此,需要确保锁的对象是同一个实例。
- 事务管理
- 如果将业务封装在A方法并在A方法使用锁,在B方法调用A方法的情况下,可能会在线程1释放锁之后提交事务的瞬间出现线程2获得锁并在线程1提交事务之前售卖库存的情况,导致超卖。因此最好在B方法调用A方法且将A整个锁住。
- 若A与B方法都在同一个类中且B调用A的情况下,需要通过手动获取动态代理对象去提交B的事务。这是因为当在同一个类中调用方法时,不会触发Spring的代理机制,导致事务注解
@Transactional无效。为了解决这个问题,可以使用AopContext.currentProxy()来获取代理对象,通过代理对象调用方法,这样事务注解才会生效。
- 使用AopContext获得代理对象
- 需要在启动类上添加
@EnableAspectJAutoProxy(exposeProxy = true)注解,以启用AspectJ自动代理并暴露代理对象。
通过以上分析可以看出,悲观锁和乐观锁各有优劣。在实际应用中,需要根据具体场景选择合适的解决方案。对于库存扣减等高并发场景,通常推荐使用悲观锁以确保数据一致性。
热门推荐
唇膏的选择与使用技巧,打造完美唇妆
正念8步法治疗焦虑
梦到鬼压身自己反抗预示着什么
新高考45个志愿填报顺序!附填写样表模板(各地汇总)
如何理解黄金递延手续费的结构与影响?这种费用对投资回报有何影响?
中国医学四大经典《黄帝八十一难经》 · 四难
睡前7个拉伸动作,赶走腰酸背痛、提升睡眠质量
碳酸钠和碳酸氢钠的pH值解析
南方冬天穿什么衣服合适?
糖尿病的饮食治疗
萨尔瓦多推进比特币革命:8万公务员将领取比特币工资
红楼梦》二十八回解读:蒋玉菡情赠茜香罗,薛宝钗羞笼红麝串
A股三大股指小幅调整:通信板块领跌,银行、电力板块逆市走强
红旗渠精神的主要内涵
皮肤松弛症如何护理
AI设计“纳米笼”模拟病毒复杂结构,可递送治疗基因并成为医疗创新平台
晚饭后饮用茶水对身体健康的影响与益处
债权转让通知回执的法律上的必要性
如何准确判断债权转让的有效性?这种判断的依据是什么?
春分的智慧:从自然到生活,寻找平衡之道
春日特调:自制花草茶与水果冰沙,把春天喝进嘴里
岁岁春无事,相逢总玉颜。30句生日祝福诗句,送给最特别的你
西梅糖尿病能吃吗
如何判断自己的内存能不能超频使用
多伦多大学医学院研究生申请条件及留学指南
多伦多大学医学专业详解:专业设置与优势
我喜欢春天因为什么一年级句子?分享适合一年级的春天描写句子
体重忽上忽下难控制?专业解析原因及解决方案
追求健康一定要先减重吗?
执业兽医师的建议:猪场设计这样做,未来省心又高效!