【Redis笔记】Redis优化秒杀

秒杀案例介绍

秒杀需要校验优惠券库存、每个用户只能使用一个优惠券,校验完成后下单。常规流程需要加锁,并发量低、数据库压力大。

如何优化?

将秒杀流程分为两个阶段,并且使用redis

第一阶段

校验优惠券库存、校验用户是否已使用优惠券。校验通过后扣减库存、记录使用优惠券的用户id。整个操作应该保证原子性,所以使用lua脚本。程序调用lua脚本,返回0则说明秒杀成功,将相关信息(优惠券id、用户id、订单id)存入阻塞队列

Redis存储

  • 优惠券库存,string类型,如下:
key:stock:vid:7
value:100

key表示id为7的优惠券,value表示库存100

  • 记录使用优惠券的用户,set类型,如下:
key:order:vid:7
value:1,2,3,4

key表示id为7的优惠券,value表示用户id集合

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.判断库存是否充足 get stockKey
if(tonumber(redis.call('get', stockKey)) <= 0) then
    -- 3.2.库存不足,返回1
    return 1
end
-- 3.2.判断用户是否下单 SISMEMBER orderKey userId
if(redis.call('sismember', orderKey, userId) == 1) then
    -- 3.3.存在,说明是重复下单,返回2
    return 2
end
-- 3.4.扣库存 incrby stockKey -1
redis.call('incrby', stockKey, -1)
-- 3.5.下单(保存用户)sadd orderKey userId
redis.call('sadd', orderKey, userId)
-- 3.6.发送消息到队列中, XADD stream.orders * k1 v1 k2 v2 ...
redis.call('xadd', 'stream.orders', '*', 'userId', userId, 'voucherId', voucherId, 'id', orderId)
return 0

第二阶段

开启后台线程监听阻塞队列,创建订单、优惠券库存扣减、优惠券用户记录等存库处理。

posted @ 2020-07-09 19:39  .Neterr  阅读(497)  评论(0编辑  收藏  举报