当前位置: 首页 > ds >正文

SpringBoot 商城系统高并发引起的库存超卖库存问题 乐观锁 悲观锁 抢购 商品秒杀 高并发

介绍

在高并发场景下,特别是商品秒杀、抢购等情况下,库存超卖问题是一个常见且棘手的问题。为了解决这个问题,Spring Boot 常使用乐观锁和悲观锁来保证数据的正确性和一致性。

在这里插入图片描述

悲观锁

悲观锁假设在多线程或多进程环境中,资源会被其他线程频繁修改,因此会尽量避免冲突,通常会在访问共享资源时加锁,直到操作完成才释放锁。这样其他线程在此期间无法访问该资源。

  • 锁粒度大:悲观锁通常是在访问资源前就加锁,直到操作完成后才解锁。
  • 性能较低:由于锁的存在,可能会导致线程等待时间较长,影响系统的并发性能。
  • 适用于资源冲突较多的场景,如银行账户转账操作。

乐观锁

乐观锁则假设在多线程环境下,资源冲突的概率较低,因此不直接加锁,而是允许多个线程并发访问共享资源。乐观锁通常依赖于数据版本控制或冲突检测来避免数据冲突。只有在提交操作时,才会检查是否有冲突发生,如果没有冲突,则提交成功;如果有冲突,则放弃或重试。

  • 锁粒度小:只有在提交数据时才进行冲突检测。
  • 性能较高:因为没有频繁加锁,线程可以并行处理多个操作,提高了系统的并发能力。
  • 适用于冲突较少的场景,如读取多、写入少的系统。

实现方式

版本号机制
乐观锁中常用的一种方式,它通过在每一条数据记录中添加一个版本号字段来判断数据是否被修改。

时间戳机制
它通过记录每条数据的最后更新时间(时间戳),来判断数据是否被修改过。


无锁测试

50个库存,200线程测试高并发。
在这里插入图片描述
超卖出40件。
在这里插入图片描述


悲观锁测试

   private final GoodsMapper goodsMapper;@Override@SneakyThrowspublic int secKillGood(int goodId)   {synchronized(this) {  // 悲观加锁,确保同一时刻只有一个线程进入该代码块,其他进来线程会被卡Goods goods =goodsMapper.selectById(goodId);if(goods.getStock()<1){return 0;}LambdaUpdateWrapper<Goods> goodsLambdaQueryWrapper =new LambdaUpdateWrapper<>();goodsLambdaQueryWrapper.eq(Goods::getId,goodId).setSql("stock = stock-1");return   goodsMapper.update(null,goodsLambdaQueryWrapper);}}

解决了超卖问题,但是在高并发的场景下如果耗时比较久,其他的请求将会被卡死,直到该锁被释放。性能低下。
在这里插入图片描述


乐观锁

查询商品库存信息和版本号 -》 判断库存是否大于0 -》 扣减库存 AND 版本号等于= 刚查询出来的版本号
如果版本号一直更新成功了,负责就是被其他线程抢了,AND条件不成立,更新失败。

在商城系统中可以使用 库存 来代替 版本号

@Override
@SneakyThrows
public int secKillGood(int goodId)   {Goods goods =goodsMapper.selectById(goodId);if(goods.getStock()<1){return 0;}LambdaUpdateWrapper<Goods> goodsLambdaQueryWrapper =new LambdaUpdateWrapper<>();goodsLambdaQueryWrapper.eq(Goods::getId,goodId).setSql("stock = stock-1") //扣减库存.eq(Goods::getStock,goods.getStock()).gt(Goods::getStock,0); //扣减时库存大于1 还有库存return   goodsMapper.update(null,goodsLambdaQueryWrapper);}

在这里插入图片描述

尽管解决了超卖的问题,但是在高并发的场景下需要频繁的访问数据还会导致性能不足,在数据量特别大的情况下应该使用其他的中间件来降低数据的增删改查。

http://www.xdnf.cn/news/7664.html

相关文章:

  • 机械安全标准示例说明
  • 机器学习算法-聚类K-Means
  • 园区无人机智能巡检项目方案
  • 机器学习自然语言处理
  • SAP ECC即将停止支持,CIO如何应对S/4HANA的迁移挑战?
  • 【机器学习】logistic回归
  • Java 02入门,封装继承多态
  • 随记1-LLM多轮对话的陷阱
  • 操作系统学习笔记第5章 (竟成)
  • [解决方案] Word转PDF
  • Android SharedPreferences:从零到一的全面解析与实战指南
  • win10使用nginx做简单负载均衡测试
  • 省赛备考~全国青少年信息素养大赛-图形化编程复赛/省赛-模拟题-判断质数合数
  • JavaScript 数组方法详解:全面指南
  • 如何优化前端应用中的JavaScript执行效率?
  • 【LinkedList demo 内部类讲说】
  • BI是什么意思?一文讲清BI的概念与应用!
  • LeetCode-前缀和-和为K的子数组
  • 网络学习中通信方面的相关知识、及再次解读B=2W
  • 如果电路教材这么讲--积分运算电路中反馈电容并联电阻的作用
  • 制造业或跨境电商相关行业三种模式:OEM、ODM、OBM
  • 十大排序算法--快速排序
  • VitePress 中以中文字符结尾的字体加粗 Markdown 格式无法解析
  • 颠覆传统:PROFINET转EthernetIP在油墨生产线的成功应用
  • 小土堆pytorch--神经网路-卷积层池化层
  • 时尚外观+专业性能丨特伦斯V30Pro重新定义便携电子钢琴
  • 深入剖析Zynq AMP模式下CPU1中断响应机制:从原理到创新实践
  • 【八股战神篇】Java虚拟机(JVM)高频面试题
  • Spring Validation校验
  • 吃透 Golang 基础:数据结构之数组