0

0

Java如何实现一个简单的商品秒杀逻辑演示_原子类与并发控制

P粉602998670

P粉602998670

发布时间:2026-02-28 09:09:18

|

894人浏览过

|

来源于php中文网

原创

synchronized无法解决分布式秒杀超卖问题,因其仅作用于单jvm实例;高并发需redis+lua或数据库行锁+乐观锁;atomicinteger仅保障单机原子读写,不支持“查-判-减”复合操作。

java如何实现一个简单的商品秒杀逻辑演示_原子类与并发控制

秒杀库存扣减为什么不能只用 synchronized

因为 synchronized 锁的是 JVM 内的某个对象,而秒杀场景下请求通常分散在多个实例(比如多台服务器),单机锁根本拦不住并发超卖。哪怕你本地压测没问题,一上生产集群就崩。

实操建议:

Logo Galleria
Logo Galleria

免费在线AI Logo生成工具,打造定制Logo

下载
  • 别把 synchronized(this)synchronized(StockService.class) 当成并发安全的万能解——它只管得住当前 JVM 进程
  • 高并发秒杀必须依赖分布式协调机制,比如 Redis + Lua 原子操作,或数据库行锁 + 乐观锁
  • 如果只是单机 Demo 演示原子类能力,可以用 AtomicInteger 模拟库存,但得明确标注“仅限学习,不可用于真实秒杀”

AtomicInteger 在秒杀中能做什么、不能做什么

AtomicInteger 能保证单个 JVM 内对一个整数变量的读-改-写是原子的,比如 decrementAndGet() 不会丢失更新。但它不提供“检查后扣减”的复合原子性——也就是典型的“先查库存是否 >0,再扣减”这种两步操作,中间仍可能被其他线程插队。

常见错误现象:get() 返回 1,两个线程同时通过判断,接着都调用 decrementAndGet(),结果变成 -1。

立即学习Java免费学习笔记(深入)”;

实操建议:

  • compareAndSet(expected, updated) 手动实现带条件的原子更新,例如:
    int current;  
    do {  
        current = stock.get();  
        if (current <= 0) break;  
    } while (!stock.compareAndSet(current, current - 1));
  • 注意 compareAndSet 是乐观策略,失败需重试,高争用下可能循环多次,不适合超大并发
  • 别用 getAndDecrement() 替代判断——它无脑减,不检查业务逻辑前提

为什么秒杀里 volatile 不能替代原子类

volatile 只保证可见性和禁止指令重排,不保证操作的原子性。对 int 类型的读写本身是原子的,但 count++ 是读+加+写三步,volatile 完全拦不住中间被截断。

使用场景:适合做状态标记,比如 volatile boolean isStarted 表示秒杀是否开启;不适合做库存计数。

实操建议:

  • volatile int stock 改成 AtomicInteger stock,否则所有递减逻辑都是假并发安全
  • 不要以为加了 volatile 就能放心写 stock-- —— 编译器不会帮你合成原子操作
  • 如果真要用 volatile 配合其他机制(比如 CAS 循环),那底层还得靠 Unsafe.compareAndSwapInt 这类支持,不是靠关键字本身

Redis + Lua 实现真正可用的秒杀扣库存(单机 Demo 级)

这才是贴近实际的轻量级方案:利用 Redis 单线程执行 Lua 脚本的特性,把“查库存、扣减、记录订单”打包成一个原子操作。Java 层只负责发请求,不操心并发控制。

示例 Lua 脚本(保存为 seckill.lua):

local stock = redis.call('GET', KEYS[1])  
if not stock or tonumber(stock) <= 0 then  
  return -1  
end  
redis.call('DECR', KEYS[1])  
return tonumber(stock) - 1

Java 调用要点:

  • RedisTemplate.execute() 加载并执行脚本,传入 Arrays.asList("seckill:stock:123")
  • 返回值为 -1 表示库存不足,非负数表示扣减后的剩余量
  • 别在 Lua 里做耗时操作(比如 HTTP 请求、复杂计算),会阻塞整个 Redis
  • 注意 Redis 持久化配置和内存淘汰策略,避免秒杀后库存被误清

复杂点在于:一旦引入 Redis,就得考虑连接池配置、超时设置、脚本加载失败降级逻辑。很多人卡在“本地跑通了,上线发现 Redis 响应慢,直接拖垮整个服务”。这不是原子类能绕过去的坎。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
什么是分布式
什么是分布式

分布式是一种计算和数据处理的方式,将计算任务或数据分散到多个计算机或节点中进行处理。本专题为大家提供分布式相关的文章、下载、课程内容,供大家免费下载体验。

401

2023.08.11

分布式和微服务的区别
分布式和微服务的区别

分布式和微服务的区别在定义和概念、设计思想、粒度和复杂性、服务边界和自治性、技术栈和部署方式等。本专题为大家提供分布式和微服务相关的文章、下载、课程内容,供大家免费下载体验。

249

2023.10.07

java中boolean的用法
java中boolean的用法

在Java中,boolean是一种基本数据类型,它只有两个可能的值:true和false。boolean类型经常用于条件测试,比如进行比较或者检查某个条件是否满足。想了解更多java中boolean的相关内容,可以阅读本专题下面的文章。

365

2023.11.13

java boolean类型
java boolean类型

本专题整合了java中boolean类型相关教程,阅读专题下面的文章了解更多详细内容。

40

2025.11.30

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

200

2023.11.20

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

200

2023.11.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

870

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

592

2024.08.29

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

0

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.9万人学习

C# 教程
C# 教程

共94课时 | 10.3万人学习

Java 教程
Java 教程

共578课时 | 73.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号