Redis实现消息队列:list、PubSub、Stream,基于Stream的异步秒杀

消息队列介绍

List实现消息队列 

Redis的list数据结构是一个双向链表,先进先出,可以利用LPUSH、LPOP、RPUSH、RPOP命令来实现元素的左右进出,但是list并不是阻塞队列,当list中无元素时,线程并不会阻塞,而是从list中取出一个null,这并不符合我们的业务需要!因此这里要用BRPOP或BLPOP命令实现阻塞效果
 

阻塞队列
当线程尝试从队列中获取元素时,若阻塞队列中无元素,则线程会阻塞,直到队列中有元素线程才会被唤醒并从阻塞队列中取出元素。

PubSub实现消息队列

Stream实现消息队列

Stream的消费者组

Redis消息队列总结

 基于Stream的异步秒杀

Redis创建消费者、消费者组、消息队列

xadd s1 * k1 v1
xgroup create s1 g1 0

xgroup create stream.orders g1 0 mkstream
XREADGROUP GROUP g1 c1 count 1 Block 2000 streams s1 >

修改seckill.lua脚本

-- 1.参数列表
-- 1.1优惠券id
local voucherId=ARGV[1]

--1.2 用户id
local userId=ARGV[2]

--1.3 订单id
local orderId=ARGV[3]

--2.数据key  ..是拼接符号
--2.1 库存key
local stockKey='seckill:stock:'..voucherId
--2.2 订单key
local orderKey='seckill:order:'..voucherId

--3.脚本业务
--3.1 判断库存是否充足
if (tonumber(redis.call('get',stockKey))<=0) then
    return 1
end
--3.2判断用户是否下单   若set集合中存在该用户id,则说明已下过单,返回1
if (tonumber(redis.call('sismember',orderKey,userId))==1) then
    return 2
end

--3.4扣库存
redis.call('incrby',stockKey,-1)
--3.5保存用户到set
redis.call('sadd',orderKey,userId)
--3.6发送消息到消息队列
redis.call('xadd','stream.orders','*','userId',userId,'voucherId',voucherId,'id',orderId)
return 0

添加新的秒杀券 

数据库和redis同时存入id为19的秒杀券

Apifox模拟用户抢购秒杀券 

查看Redis中数据变化

1、Redis中id=19的秒杀券的数量都-1

 

2、Redis中成功存入用户id 

3、 Redis中成功存入订单信息

 4、数据库中成功存入订单信息

总结 

VoucherOrderServiceImpl.java完整代码

@Slf4j
@Service
public class VoucherOrderServiceImpl extends ServiceImpl<VoucherOrderMapper, VoucherOrder>  implements IVoucherOrderService {

    @Resource
    private StringRedisTemplate stringRedisTemplate;
    @Resource
    private ISeckillVoucherService SeckillVoucherService;

    @Autowired
    private RedissonClient redissonClient;

    @Resource
    private RedisIdWorker redisIdWorker;

    //阻塞队列  当线程尝试从队列中获取元素时,若队列无元素,则线程会阻塞,直到队列中有元素才会被唤醒
//    private BlockingQueue<VoucherOrder> orderTasks=new ArrayBlockingQueue<>(1024*1024);

    //线程池,负责从阻塞队列中获取订单然后异步下单
    private static final ExecutorService SECKILL_ORDER_EXECUTOR= Executors.newSingleThreadExecutor();

    //spring提供的注解  作用:类初始化后就执行VoucherOrderHandler方法
    //向线程池提交一个线程
    @PostConstruct
    private void init(){
        SECKILL_ORDER_EXECUTOR.submit(new VoucherOrderHandler());
    }

    //线程  内部类
    private class VoucherOrderHandler implements Runnable{

        String queueName ="stream.orders";
        @Override
        public void run() {
            while (true){
                try {
                    //获取队列中的订单信息
                    //  VoucherOrder order = orderTasks.take();

                    //获取消息队列的订单信息
                    List<MapRecord<String, Object, Object>> list = stringRedisTemplate.opsForStream().read(Consumer.from("g1", "c1")
                            , StreamReadOptions.empty().count(1).block(Duration.ofSeconds(2))
                            , StreamOffset.create(queueName, ReadOffset.lastConsumed())
                    );
                    //判断消息是否获取成功
                    if (list==null||list.isEmpty()){
                        //获取失败说明没有消息,继续循环
                        continue;
                    }
                    //获取成功则创建订单
                    MapRecord<String, Object, Object> record = list.get(0);
                    Map<Object, Object> value = record.getValue();
                    VoucherOrder order = BeanUtil.fillBeanWithMap(value, new VoucherOrder(), true);
                    createVoucherOrder(order);
                     //ACK确认信息
                    stringRedisTemplate.opsForStream().acknowledge("s1","g1",record.getId());
                } catch (Exception e) {
                    log.error("订单处理异常",e);
                    handlePendingList();
                }
            }
        }
        private void handlePendingList() {
            while (true){
                try {
                    //获取pendingList队列的订单信息
                    List<MapRecord<String, Object, Object>> list = stringRedisTemplate.opsForStream().read(
                            Consumer.from("g1", "c1")
                            , StreamReadOptions.empty().count(1)
                            , StreamOffset.create(queueName, ReadOffset.from("0"))
                    );
                    //判断消息是否获取成功
                    if (list==null||list.isEmpty()){
                        //获取失败说明没有消息,结束循环
                        break;
                    }

                    //获取成功则下单
                    MapRecord<String, Object, Object> record = list.get(0);
                    Map<Object, Object> value = record.getValue();
                    VoucherOrder order = BeanUtil.fillBeanWithMap(value, new VoucherOrder(), true);
                    createVoucherOrder(order);

                    //ACK确认信息
                    stringRedisTemplate.opsForStream().acknowledge("s1","g1",record.getId());
                } catch (Exception e) {
                    log.error("pendingList处理异常",e);

                }

            }
        }
    }

    //代理对象
    IVoucherOrderService proxy;

    private void handleVoucherOrder(VoucherOrder order) {

        Long userId = order.getUserId();

        RLock redisLock = redissonClient.getLock("lock:order:" + userId);

        boolean tryLock = redisLock.tryLock();

        //判断锁是否获取成功
        if (!tryLock){
            log.error("不允许重复下单");
            return ;
        }
        try {
            //锁加到这里,事务提交后才释放锁
//            proxy.createVoucherOrder(order);
            //使用动态代理类的对象,事务可以生效
        } finally {
            redisLock.unlock();
        }
    }

    private static final DefaultRedisScript<Long> SECKILL_SCRIPT;
    static  {
        SECKILL_SCRIPT=new DefaultRedisScript<>();
        SECKILL_SCRIPT.setLocation(new ClassPathResource("seckill.lua"));
        SECKILL_SCRIPT.setResultType(Long.class);
    }

    @Override
    public Result seckillVoucher(Long voucherId) {
        Long userId = UserHolder.getUser().getId();

        //TODO 生成了orderId,把订单信息保存到阻塞队列,由另一个线程专门根据订单信息去数据库做增删改查,这就实现了异步
        long orderId = redisIdWorker.nextId("order");

        //执行lua脚本判断有无购买资格
        Long result = stringRedisTemplate.execute(
                SECKILL_SCRIPT,
                Collections.emptyList(),
                voucherId.toString(), userId.toString(),String.valueOf(orderId)
        );
        int i = result.intValue();
        if (i!=0){
            return Result.fail(i==1?"优惠券库存不足":"不能重复下单");
        }


        //获取事务的动态代理对象,需要在启动类加注解暴漏出对象
        proxy = (IVoucherOrderService)AopContext.currentProxy();//拿到动态代理对象

        //添加到阻塞队列
        //orderTasks.add(order);

        return Result.ok(orderId);
    }

    //TODO spring对该类做了动态代理,用动态代理的对象提交的事务
    @Transactional
    public void createVoucherOrder(VoucherOrder order) {
        //一人一单,根据优惠卷id和用户id去数据库查询是否已经存在该优惠卷
        Long id = order.getUserId();
        Long userId = order.getUserId();

        RLock redisLock = redissonClient.getLock("lock:order:" + userId);
        boolean isLock = redisLock.tryLock();
        if (!isLock){
            log.error("不允许重复下单");
            return;
        }
            //为用户id加锁而不是对整个createVoucherOrder方法加锁,减小锁范围,提升性能,这样每个用户就有不同的锁
            //锁加在函数内部,锁内的代码执行完后就会释放锁,而事务的提交是在整个方法执行后提交的,也就是事务的提交在锁释放之后。
            //但是锁释放后其他线程就可以进来,此时事务可能还没有提交,可能出现并发问题,重复购买
            //所以要扩大锁的范围,把锁加到seckillVoucher方法后面,在事务提交后才能释放锁!

        try {
            int count = query().eq("user_id", id).eq("voucher_id", order.getVoucherId()).count();
            if (count >=1) {
                //count==1说明用户拥有了一个优惠券
                log.error("不能重复下单");
                return;
//                return Result.fail("不能重复购买优惠卷");
            }

            //4.扣减库存  防止超卖,加乐观锁,扣减库存前再查询一次库存判断
//        boolean b = SeckillVoucherService.update()
//                .setSql("stock=stock-1").
//                eq("voucher_id", voucherId).eq("stock",voucher.getStock()).update();
            //使用setSql方法设置了更新语句"stock=stock-1",接着使用eq方法添加了两个条件:"voucher_id"等于voucherId和"stock"等于voucher.getStock()
            //条件1:voucher_id=voucherId指当前操作的优惠卷的id=数据库中的优惠卷id,即通过优惠卷id指明了要修改哪个优惠卷的库存
            //条件2:stock=voucher.getStock,说明该线程修改库存期间没有其他线程来插队修改库存,那么数据是安全的
            //TODO !!!注意!这种操作在并发情况下可能导致用户在优惠卷库存充足的情况下抢购优惠卷失败,也就是即使有库存也会抢购失败,此时可以判断库存是否充足,重新抢购

            //修改如下:最后库存判断,只要>0就可以修改
            boolean b = SeckillVoucherService.update()
                    .setSql("stock=stock-1").
                    eq("voucher_id", order.getVoucherId()).gt("stock", 0)
                    .update();

            if (!b) {
//                return Result.fail("库存不足");
                log.error("库存不足");
                return;
            }
            //创建订单
            save(order);
        } finally {
            //释放锁
            redisLock.unlock();
        }
    }
}

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值