Redis实现限量优惠券的秒杀功能

 更新时间:2024年12月03日 10:18:25   作者:mxbb.  
文章详细分析了避免超卖问题的方法,包括确保一人一单的业务逻辑,并提供了代码实现步骤和代码示例,感兴趣的朋友跟随小编一起看看吧

核心:避免超卖问题,保证一人一单  业务逻辑

代码步骤分析

全部代码

@Service
public class VoucherOrderServiceImpl extends ServiceImpl<VoucherOrderMapper, VoucherOrder> implements IVoucherOrderService {
    @Resource
    private ISeckillVoucherService seckillVoucherService;
    @Resource
    private RedisIdWorker redisIdWorker;
    @Resource
    private StringRedisTemplate stringRedisTemplate;
    @Resource
    private RedissonClient redissonClient;
    @Override
    public 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();
        }
        //}//事务提交之后在释放锁
    }
    /**
     * 这里对于加锁做出一些解释:如果锁加在整个方法上,那么就会导
     *  致锁的粒度过大,导致每个进程进来都会锁住,所以要控制锁的粒度
     */
    @Transactional
    public 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 订单ID
        VoucherOrder voucherOrder = new VoucherOrder();
        long orderid = RedisIdWorker.nextId("order");
        voucherOrder.setVoucherId(orderid);
        //6.2 用户ID
        voucherOrder.setUserId(userId);
        //6.3 代金券ID
        voucherOrder.setVoucherId(voucherId);
        save(voucherOrder);
        //7 返回订单id
        return Result.ok(orderid);
    }
}

到此这篇关于Redis实现限量优惠券的秒杀功能的文章就介绍到这了,更多相关Redis优惠券秒杀内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Redis数据结构-跳跃表skiplist详解

    Redis数据结构-跳跃表skiplist详解

    这篇文章主要介绍了Redis数据结构-跳跃表skiplist,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2025-09-09
  • 基于Redis 实现网站PV/UV数据统计

    基于Redis 实现网站PV/UV数据统计

    PV和UV是两个重要的指标,本文主要介绍了基于Redis 实现网站PV/UV数据统计,具有一定的参考价值,感兴趣的可以了解一下
    2025-04-04
  • redis和redission分布式锁原理及区别说明

    redis和redission分布式锁原理及区别说明

    文章对比了synchronized、乐观锁、Redis分布式锁及Redission锁的原理与区别,指出在集群环境下synchronized失效,乐观锁存在数据库性能瓶颈,而Redission通过watchdog自动续期和Lua原子操作解决Redis锁的超时问题,推荐其在高并发场景下的可靠性与易用性
    2025-08-08
  • Redis拒绝连接问题分析与解决方案

    Redis拒绝连接问题分析与解决方案

    在分布式系统中,Redis作为高性能的内存数据库,广泛用于缓存、消息队列、会话管理等场景,然而,随着系统复杂度和并发量的增加,Redis连接问题时有发生,尤其是"拒绝连接"的错误,本文将深入分析Redis拒绝连接的常见原因,并详细讲解每种原因的解决方案
    2024-10-10
  • 浅谈Redis如何应对并发访问

    浅谈Redis如何应对并发访问

    本文主要介绍了Redis如何应对并发访问,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-08-08
  • Redis高可用的三种实现方式

    Redis高可用的三种实现方式

    在实际生产环境中为保证Redis的服务连续性和可靠性,需要设计一个高可用架构,本文就来介绍一下Redis高可用的三种实现方式,主要包括主从复制模式,Redis Sentinel模式和Redis Cluster模式,感兴趣的可以了解一下
    2023-12-12
  • Redis中过期键如何删除示例详解

    Redis中过期键如何删除示例详解

    因为redis数据是基于内存的,然而内存是非常宝贵的资源,然后我们就会对一些不常用或者只用一次的数据进行存活时间设置,这样才能提高内存的使用效率,下面这篇文章主要给大家介绍了关于Redis中过期键如何删除的相关资料,需要的朋友可以参考下
    2022-04-04
  • Redis List列表的详细介绍

    Redis List列表的详细介绍

    这篇文章主要介绍了Redis List列表的详细介绍的相关资料,Redis列表是简单的字符串列表,按照插入顺序排序,需要的朋友可以参考下
    2017-08-08
  • Redis慢查询日志及慢查询分析详解

    Redis慢查询日志及慢查询分析详解

    这篇文章主要为大家介绍了Redis慢查询日志及慢查询分析详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-01-01
  • redis restore 命令的用法实例详解

    redis restore 命令的用法实例详解

    Redis的RESTORE命令用于将DUMP生成的序列化数据恢复为键值,适用于数据迁移、备份恢复和跨实例同步,本文介绍redis restore 命令的用法,感兴趣的朋友一起看看吧
    2025-02-02

最新评论