LUA+Reids实现库存秒杀预扣减 记录流水 以及自己的思考

目录

lua脚本

记录流水

记录流水的作用

流水什么时候删除


我们在做库存扣减的时候,显示基于Lua脚本和Redis实现的预扣减

这样可以在秒杀扣减的时候保证操作的原子性和高效性

lua脚本

复制代码
// ... 已有代码 ...

    @Override
    public InventoryResponse decrease(InventoryRequest request) {
        // 创建库存响应对象
        InventoryResponse inventoryResponse = new InventoryResponse();
        // 定义用于减少库存的Lua脚本
        String luaScript = """
                -- 检查哈希表 KEYS[2] 中是否已存在 ARGV[2] 对应的字段
                -- 如果存在,说明该操作已执行过,返回错误信息
                if redis.call('hexists', KEYS[2], ARGV[2]) == 1 then
                    return redis.error_reply('OPERATION_ALREADY_EXECUTED')
                end
                                
                -- 从 Redis 中获取 KEYS[1] 对应的值,即当前库存
                local current = redis.call('get', KEYS[1])
                -- 如果返回值为 false,说明键不存在,返回错误信息
                if current == false then
                    return redis.error_reply('KEY_NOT_FOUND')
                end
                -- 尝试将当前库存值转换为数字,如果转换失败,返回错误信息
                if tonumber(current) == nil then
                    return redis.error_reply('current value is not a number')
                end
                -- 如果当前库存为 0,返回库存为零的错误信息
                if tonumber(current) == 0 then
                    return redis.error_reply('INVENTORY_IS_ZERO')
                end
                -- 如果当前库存小于要减少的数量 ARGV[1],返回库存不足的错误信息
                if tonumber(current) < tonumber(ARGV[1]) then
                    return redis.error_reply('INVENTORY_NOT_ENOUGH')
                end
                                
                -- 计算减少库存后的新库存值
                local new = tonumber(current) - tonumber(ARGV[1])
                -- 将新的库存值存储回 Redis 中
                redis.call('set', KEYS[1], tostring(new))
                                
                -- 获取 Redis 服务器的当前时间(秒和微秒)
                local time = redis.call("time")
                -- 将获取到的时间转换为毫秒级时间戳
                local currentTimeMillis = (time[1] * 1000) + math.floor(time[2] / 1000)
                                
                -- 使用哈希结构存储库存减少操作的日志
                -- 在 KEYS[2] 对应的哈希表中,以 ARGV[2] 为字段名,存储操作日志的 JSON 字符串
                redis.call('hset', KEYS[2], ARGV[2], cjson.encode({
                    action = "decrease",  -- 操作类型为减少库存
                    from = current,       -- 操作前的库存值
                    to = new,             -- 操作后的库存值
                    change = ARGV[1],     -- 减少的库存数量
                    by = ARGV[2],         -- 操作标识
                    timestamp = currentTimeMillis  -- 操作的时间戳
                }))
                                
                -- 返回更新后的库存值
                return new
                """;

        try {
            // ... 已有代码 ...

首先是合法性校验

然后是原子性扣减

最后记录一条库存扣减流水

记录流水

在 lua 脚本中 我们不仅是做了库存的扣减

还在 redis 中用 hash 存储了一条流水

key:买家id + token + 扣减数量

value:本次扣减的变化的库存数,变化前的库存数,变化后的库存数 变化操作的ID 变化的时间戳

记录流水的作用

  1. 幂等

执行lua脚本的时候 查看是否有当前流水

如果有 说明是一个重复请求 直接幂等掉

  1. 对账

很重要

redis库存扣减后 需要持久化到数据库 这边使用的是mq异步处理

如何保证一致性呢

我们用redis里的hash流水和数据库流水进行对账

如果不一致的话

很有可能是mq丢消息了

就要会进行排查了

流水什么时候删除

数据库 与 缓存对账后删除

商品下架后24小时后删除

相关推荐
Maple_land5 分钟前
Linux进程第八讲——进程状态全景解析(二):从阻塞到消亡的完整生命周期
linux·运维·服务器·c++·centos
嵌入式分享7 分钟前
嵌入式分享#41:RK3576改UART波特率【精简版】
linux·嵌入式硬件·ubuntu·嵌入式
爱吃生蚝的于勒9 分钟前
【Linux】零基础学会Linux之权限
linux·运维·服务器·数据结构·git·算法·github
咖啡续命又一天15 分钟前
python 自动化采集 ChromeDriver 安装
开发语言·python·自动化
惜.己17 分钟前
linux中jenkins正常启动外部无法访问
linux·servlet·jenkins
Cyan_RA928 分钟前
Linux 远程Ubuntu服务器本地部署大模型 EmoLLM 中常见的问题及解决方案 万字详解
linux·运维·服务器·ubuntu·大模型·远程部署·emollm
阿湯哥31 分钟前
Redis数据库隔离业务缓存对查询性能的影响分析
数据库·redis·缓存
麦兜*32 分钟前
Redis 7.2 新特性实战:Client-Side Caching(客户端缓存)如何大幅降低延迟?
数据库·spring boot·redis·spring·spring cloud·缓存·tomcat
minji...40 分钟前
Linux相关工具vim/gcc/g++/gdb/cgdb的使用详解
linux·运维·服务器·c++·git·自动化·vim
web安全工具库1 小时前
Linux 高手进阶:Vim 核心模式与分屏操作详解
linux·运维·服务器·前端·数据库