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

Redis实现限量优惠券的秒杀

核心:避免超卖问题,保证一人一单 

业务逻辑

 

代码步骤分析

全部代码

@Service
public class VoucherOrderServiceImpl extends ServiceImpl<VoucherOrderMapper, VoucherOrder> implements IVoucherOrderService {@Resourceprivate ISeckillVoucherService seckillVoucherService;@Resourceprivate RedisIdWorker redisIdWorker;@Resourceprivate StringRedisTemplate stringRedisTemplate;@Resourceprivate RedissonClient redissonClient;@Overridepublic Result seckillVoucher(Long voucherId) {//1 查询优惠券信息SeckillVoucher seckillVoucher = seckillVoucherService.getById(voucherId);//2 判断秒杀是否开始LocalDateTime now = LocalDateTime.now(); //现在时间LocalDateTime beginTime = seckillVoucher.getBeginTime(); //开始时间LocalDateTime endTime = seckillVoucher.getEndTime(); //结束时间if (now.isBefore(beginTime)) {return Result.fail("秒杀还未开始");}//3 判断秒杀是否结束if (now.isAfter(endTime)) {return Result.fail("秒杀已结束");}//4 判断库存是否充足int stock = (int) seckillVoucher.getStock();if (stock == 0 && stock <= 0) {return Result.fail("库存不足");}//确保一人一单 用户ID和代金券ID联合查询Long userId = UserHolder.getUser().getId();//先获取锁 再提交事务//synchronized (userId.toString().intern()) {//创建锁对象//SimpleRedisLock simpleRedisLock = new SimpleRedisLock("order:" + userId, stringRedisTemplate);RLock lock = redissonClient.getLock("lock:order:" + userId);//获取锁boolean isLock = lock.tryLock();//判断是否获取锁成功if (!isLock){//获取锁失败 返回错误或失败return Result.fail("不允许重复下单");}try {//获取事务的代理对象IVoucherOrderService proxy = (IVoucherOrderService) AopContext.currentProxy();return proxy.creatVoucherOrder(voucherId);} finally {//释放锁lock.unlock();}//}//事务提交之后在释放锁}/*** 这里对于加锁做出一些解释:如果锁加在整个方法上,那么就会导*  致锁的粒度过大,导致每个进程进来都会锁住,所以要控制锁的粒度*/@Transactionalpublic Result creatVoucherOrder(Long voucherId) {Long userId = UserHolder.getUser().getId();Integer count = query().eq("user_id", userId).eq("voucher_id", voucherId).count();if (count > 0) {return Result.fail("用户已经购买过了");}//5 扣减库存-boolean success = seckillVoucherService.update().setSql("stock = stock -1 ")  //乐观锁.eq("voucher_id", voucherId).gt("stock", 0) //判断库存是否大于0: where id = ? and stock > 0.update();if (!success) {return Result.fail("库存不足");}//6 创建订单//6.1 订单IDVoucherOrder voucherOrder = new VoucherOrder();long orderid = RedisIdWorker.nextId("order");voucherOrder.setVoucherId(orderid);//6.2 用户IDvoucherOrder.setUserId(userId);//6.3 代金券IDvoucherOrder.setVoucherId(voucherId);save(voucherOrder);//7 返回订单idreturn Result.ok(orderid);}}

 

http://www.lryc.cn/news/495942.html

相关文章:

  • Linux centOS 7 安装 rabbitMQ
  • 活着就好20241202
  • 自由学习记录(28)
  • 操作系统、虚拟化技术与云原生01
  • linux的挂卸载
  • 【和春笋一起学C++】OpenCV中数组和指针运用实例
  • Maya 中创建游戏角色的头发,并将其导出到 Unreal Engine 5
  • React 路由(React Router):在 React 应用中管理路由
  • SAP-CPI组件Transformation介绍之Converter
  • Laravel 代理收益排行榜
  • LeetCode hot100面试背诵版(自用)
  • 常见的Web安全漏洞——XSS
  • liteflow 架构详解
  • 国产麒麟操作系统上运行LabVIEW
  • 【C语言】结构体(一)
  • C++《set与map》
  • 深度学习-52-AI应用实战之基于Yolo8的目标检测自动标注
  • 【Elasticsearch】05-DSL查询
  • qml项目创建的区别
  • .NET8/.NETCore 依赖注入:自动注入项目中所有接口和自定义类
  • Flutter:city_pickers省市区三级联动
  • Kafka-Connect自带示例
  • Hbase应用案例 随机号码生成
  • 论文阅读——量子退火Experimental signature of programmable quantum annealing
  • (长期更新)《零基础入门 ArcGIS(ArcMap) 》实验二----网络分析(超超超详细!!!)
  • go语言 Pool实现资源池管理数据库连接资源或其他常用需要共享的资源
  • mysql一个事务最少几次IO操作
  • 运输层总结
  • 【嵌套查询】.NET开源 ORM 框架 SqlSugar 系列
  • React 前端框架1