解决秒杀系统库存超卖问题:乐观锁与Redis分布式锁的应用

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介: 解决秒杀系统库存超卖问题:乐观锁与Redis分布式锁的应用

解决秒杀系统库存超卖问题:乐观锁与Redis分布式锁的应用

秒杀系统在高并发场景下,库存超卖问题一直是业务开发中的一大难题。本文将详细介绍如何使用乐观锁和Redis分布式锁来解决这一问题,以确保秒杀系统的稳定性和高性能。

1. 乐观锁解决方案

乐观锁是一种无锁机制,通过版本号的方式实现并发控制。在秒杀系统中,我们可以在商品表中增加一个版本号字段,每次更新库存时同时更新版本号。用户提交秒杀请求时,先获取商品的版本号,然后在更新库存时验证版本号是否仍然一致,如果一致则更新成功,否则说明有其他用户已经修改了库存。以下是具体的实现代码:

1.1 数据库表设计
CREATE TABLE t_goods (
    id INT PRIMARY KEY,
    name VARCHAR(255),
    stock INT,
    version INT
);
1.2 商品服务代码
@Service
public class GoodsService {
    @Autowired
    private GoodsDao goodsDao;
    public boolean seckillGoods(long userId, long goodsId) {
        Goods goods = goodsDao.getGoodsById(goodsId);
        if (goods.getStock() > 0) {
            int affectedRows = goodsDao.reduceStock(goodsId, goods.getVersion());
            if (affectedRows > 0) {
                // 更新成功,进行秒杀操作
                createSeckillOrder(userId, goodsId);
                return true;
            }
        }
        return false;
    }
}
1.3 数据库更新方法
@Update("UPDATE t_goods SET stock = stock - 1, version = version + 1 WHERE id = #{goodsId} AND version = #{version}")
int reduceStock(@Param("goodsId") long goodsId, @Param("version") int version);

2. Redis分布式锁解决方案

在分布式环境中,为了确保多个节点的并发安全,我们可以使用Redis分布式锁。以下是使用Redisson实现的分布式锁代码:

2.1 引入Redisson依赖
<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson-spring-boot-starter</artifactId>
    <version>3.17.1</version> <!-- 请根据实际情况选择版本 -->
</dependency>
2.2 Redis分布式锁代码
@Service
public class GoodsService {
    @Autowired
    private RedissonClient redissonClient;
    public boolean seckillGoods(long userId, long goodsId) {
        String lockKey = "seckill:" + goodsId;
        RLock lock = redissonClient.getLock(lockKey);
        try {
            if (lock.tryLock(500, 10, TimeUnit.MILLISECONDS)) {
                Goods goods = goodsDao.getGoodsById(goodsId);
                if (goods.getStock() > 0) {
                    // 秒杀成功,更新库存
                    goodsDao.reduceStock(goodsId);
                    // 创建秒杀订单等操作
                    return true;
                }
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        } finally {
            lock.unlock();
        }
        return false;
    }
}

3. 优缺点比较

3.1 乐观锁
优点:
  • 无锁机制,不会阻塞其他线程。
  • 对于读多写少的场景性能较好。
缺点:
  • 需要额外的版本号字段。
  • 在高并发写入场景下,容易出现CAS失败,需要重试。
3.2 Redis分布式锁
优点:
  • 在分布式环境中保证了并发安全。
  • 使用简单,不需要修改数据库表结构。
缺点:
  • 引入了额外的依赖,可能对性能有一定影响。
  • 锁的过期时间需要谨慎设置,过长可能导致死锁,过短可能引起并发问题。
相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
相关文章
|
6天前
|
NoSQL Java Redis
太惨痛: Redis 分布式锁 5个大坑,又大又深, 如何才能 避开 ?
Redis分布式锁在高并发场景下是重要的技术手段,但其实现过程中常遇到五大深坑:**原子性问题**、**连接耗尽问题**、**锁过期问题**、**锁失效问题**以及**锁分段问题**。这些问题不仅影响系统的稳定性和性能,还可能导致数据不一致。尼恩在实际项目中总结了这些坑,并提供了详细的解决方案,包括使用Lua脚本保证原子性、设置合理的锁过期时间和使用看门狗机制、以及通过锁分段提升性能。这些经验和技巧对面试和实际开发都有很大帮助,值得深入学习和实践。
太惨痛: Redis 分布式锁 5个大坑,又大又深, 如何才能 避开 ?
|
3天前
|
SQL 关系型数据库 MySQL
乐观锁在分布式数据库中如何与事务隔离级别结合使用
乐观锁在分布式数据库中如何与事务隔离级别结合使用
|
13天前
|
消息中间件 缓存 NoSQL
Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。
【10月更文挑战第4天】Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。随着数据增长,有时需要将 Redis 数据导出以进行分析、备份或迁移。本文详细介绍几种导出方法:1)使用 Redis 命令与重定向;2)利用 Redis 的 RDB 和 AOF 持久化功能;3)借助第三方工具如 `redis-dump`。每种方法均附有示例代码,帮助你轻松完成数据导出任务。无论数据量大小,总有一款适合你。
50 6
|
14天前
|
缓存 NoSQL Java
大数据-50 Redis 分布式锁 乐观锁 Watch SETNX Lua Redisson分布式锁 Java实现分布式锁
大数据-50 Redis 分布式锁 乐观锁 Watch SETNX Lua Redisson分布式锁 Java实现分布式锁
37 3
大数据-50 Redis 分布式锁 乐观锁 Watch SETNX Lua Redisson分布式锁 Java实现分布式锁
|
5天前
|
人工智能 文字识别 Java
SpringCloud+Python 混合微服务,如何打造AI分布式业务应用的技术底层?
尼恩,一位拥有20年架构经验的老架构师,通过其深厚的架构功力,成功指导了一位9年经验的网易工程师转型为大模型架构师,薪资逆涨50%,年薪近80W。尼恩的指导不仅帮助这位工程师在一年内成为大模型架构师,还让他管理起了10人团队,产品成功应用于多家大中型企业。尼恩因此决定编写《LLM大模型学习圣经》系列,帮助更多人掌握大模型架构,实现职业跃迁。该系列包括《从0到1吃透Transformer技术底座》、《从0到1精通RAG架构》等,旨在系统化、体系化地讲解大模型技术,助力读者实现“offer直提”。此外,尼恩还分享了多个技术圣经,如《NIO圣经》、《Docker圣经》等,帮助读者深入理解核心技术。
SpringCloud+Python 混合微服务,如何打造AI分布式业务应用的技术底层?
|
7天前
|
NoSQL Redis 数据库
计数器 分布式锁 redis实现
【10月更文挑战第5天】
18 1
|
11天前
|
NoSQL 算法 关系型数据库
Redis分布式锁
【10月更文挑战第1天】分布式锁用于在多进程环境中保护共享资源,防止并发冲突。通常借助外部系统如Redis或Zookeeper实现。通过`SETNX`命令加锁,并设置过期时间防止死锁。为避免误删他人锁,加锁时附带唯一标识,解锁前验证。面对锁提前过期的问题,可使用守护线程自动续期。在Redis集群中,需考虑主从同步延迟导致的锁丢失问题,Redlock算法可提高锁的可靠性。
36 4
|
3天前
|
SQL 监控 NoSQL
乐观锁在分布式系统中如何实现
乐观锁在分布式系统中如何实现
|
11天前
|
缓存 NoSQL 算法
面试题:Redis如何实现分布式锁!
面试题:Redis如何实现分布式锁!
|
13天前
|
缓存 网络协议 API
分布式系统应用之服务发现!
分布式系统应用之服务发现!