news 2026/6/9 3:11:47

Spring Boot 中基于线程池的订单创建并行化实践

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Spring Boot 中基于线程池的订单创建并行化实践

一、背景

1.1 业务背景

  • 以电商系统「订单创建」接口为例

  • 一个用户下单请求,往往需要完成多个业务步骤:

    • 校验库存

    • 校验用户信息

    • 计算订单价格

    • 锁库存

    • 创建订单

1.2 问题描述

  • 传统实现方式:串行执行

  • 在高并发场景下:

    • 接口 RT 高

    • 线程被长时间占用

    • 系统吞吐下降

1.3 技术挑战

  • 哪些任务可以并行?

  • 如何安全、高效地并行?

  • 如何在 Spring Boot 中正确使用线程池

  • 二、业务场景分析与并行拆分

    2.1 订单创建流程拆解

    步骤是否存在依赖是否可并行
    校验库存
    校验用户
    计算价格
    锁库存依赖库存校验
    创建订单依赖前置结果

    2.2 并行化设计思路

  • 无依赖的校验类任务 → 并行

  • 存在业务依赖的核心流程 → 串行

  • 线程池只用于短生命周期任务

三、技术选型与整体设计

3.1 为什么不直接 new Thread?

  • 线程创建成本高

  • 无法控制并发量

  • 高并发下容易导致 JVM 失控

3.2 为什么选择线程池 + CompletableFuture?

  • 线程复用,降低系统开销

  • 明确的并发上限

  • 支持任务编排(allOf / thenCombine)

3.3 在 Spring Boot 中的正确姿势

  • 线程池必须交由 Spring 管理

  • 使用ThreadPoolTaskExecutor

  • 为业务定制专用线程池,避免互相影响

四、线程池设计与配置(Core)

4.1 线程池配置代码

@Configuration public class OrderThreadPoolConfig { @Bean("orderExecutor") public ThreadPoolTaskExecutor orderExecutor() { ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); executor.setCorePoolSize(8); executor.setMaxPoolSize(16); executor.setQueueCapacity(200); executor.setKeepAliveSeconds(60); executor.setThreadNamePrefix("order-create-"); executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy()); executor.initialize(); return executor; } }

4.2 关键参数说明

  • corePoolSize:常态并发能力

  • maxPoolSize:应对突发流量

  • queueCapacity:缓冲任务,防止雪崩

  • RejectedExecutionHandler

    • 选择CallerRunsPolicy实现自然限流

4.3 线程池定位

  • Web 请求内使用

  • IO + 轻计算混合型线程池

  • 非长任务、非阻塞型任务

五、核心业务实现

5.1 校验 Service(模拟 RPC / DB)

@Service public class OrderCheckService { public boolean checkStock(Long skuId) { sleep(100); return true; } public boolean checkUser(Long userId) { sleep(80); return true; } public int calcPrice(Long skuId) { sleep(120); return 99; } private void sleep(long ms) { try { Thread.sleep(ms); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } } }

5.2 订单创建核心逻辑

@Service public class OrderService { @Resource(name = "orderExecutor") private Executor orderExecutor; @Autowired private OrderCheckService orderCheckService; public String createOrder(Long userId, Long skuId) throws Exception { long start = System.currentTimeMillis(); CompletableFuture<Boolean> stockFuture = CompletableFuture.supplyAsync( () -> orderCheckService.checkStock(skuId), orderExecutor); CompletableFuture<Boolean> userFuture = CompletableFuture.supplyAsync( () -> orderCheckService.checkUser(userId), orderExecutor); CompletableFuture<Integer> priceFuture = CompletableFuture.supplyAsync( () -> orderCheckService.calcPrice(skuId), orderExecutor); CompletableFuture.allOf( stockFuture, userFuture, priceFuture).join(); if (!stockFuture.get()) { throw new RuntimeException("库存不足"); } int price = priceFuture.get(); lockStock(skuId); saveOrder(userId, skuId, price); return "success, cost=" + (System.currentTimeMillis() - start) + "ms"; } private void lockStock(Long skuId) { sleep(50); } private void saveOrder(Long userId, Long skuId, int price) { sleep(80); } private void sleep(long ms) { try { Thread.sleep(ms); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } } }

5.3 Controller

@RestController @RequestMapping("/order") public class OrderController { @Autowired private OrderService orderService; @PostMapping("/create") public String createOrder( @RequestParam Long userId, @RequestParam Long skuId) throws Exception { return orderService.createOrder(userId, skuId); } }

六、JMeter 压测与结果分析

6.1 压测配置说明

  • 并发线程数:200

  • Ramp-Up:1 秒


6.2 压测现象

  • 系统启动初期:

    • 接口 RT ≈288ms

  • 持续压测后:

    • RT 逐渐上升

    • 峰值约1888ms


6.3 原因分析

  1. 每个请求会向线程池提交3 个并行任务

  2. 200 个并发请求 ≈600 个线程池任务

  3. 线程池最大并发执行数为16

  4. 多余任务进入阻塞队列

  5. 队列满后触发CallerRunsPolicy

  6. 部分任务由HTTP 工作线程执行,导致请求处理时间变长


6.4 工程结论

  • RT 上升并不代表线程池失效

  • 这是线程池在高并发下的自我保护行为

  • 相比无限创建线程导致系统崩溃,RT 变慢是一种可接受的退化方式

线程池优化的是系统吞吐与稳定性,而不是在无限并发下保持恒定响应时间。

七、问题思考

7.1 为什么不用 @Async?

  • 难以进行复杂任务编排

  • 不利于精细化控制线程池

7.2 为什么不用 parallelStream?

  • 使用公共 ForkJoinPool

  • 线程资源不可控

7.3 线程池并非万能

  • 仍需配合限流、熔断等机制

  • 核心链路与非核心链路应区别对待

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/1 14:14:09

TouchGal:一站式Galgame社区平台完整指南

TouchGal是专为视觉小说爱好者打造的一站式Galgame社区平台&#xff0c;提供海量游戏资源下载、深度社区交流和个性化管理工具&#xff0c;让你的Galgame体验更加丰富多彩。 【免费下载链接】kun-touchgal-next TouchGAL是立足于分享快乐的一站式Galgame文化社区, 为Gal爱好者提…

作者头像 李华
网站建设 2026/6/9 0:21:28

ServerPackCreator完整指南:快速构建Minecraft服务器包的最佳工具

ServerPackCreator完整指南&#xff1a;快速构建Minecraft服务器包的最佳工具 【免费下载链接】ServerPackCreator Create a server pack from a Minecraft Forge, NeoForge, Fabric, LegacyFabric or Quilt modpack! 项目地址: https://gitcode.com/gh_mirrors/se/ServerPac…

作者头像 李华
网站建设 2026/5/30 23:08:48

FanControl.HWInfo智能散热系统深度配置指南

实战演练&#xff1a;从零搭建智能散热环境 【免费下载链接】FanControl.HWInfo FanControl plugin to import HWInfo sensors. 项目地址: https://gitcode.com/gh_mirrors/fa/FanControl.HWInfo 环境部署三步法 第一步&#xff1a;获取核心插件组件&#xff0c;从指定…

作者头像 李华
网站建设 2026/5/27 12:40:26

WinClean:Windows系统终极优化与清理完整指南

WinClean&#xff1a;Windows系统终极优化与清理完整指南 【免费下载链接】WinClean Windows optimization and debloating utility. 项目地址: https://gitcode.com/gh_mirrors/wi/WinClean WinClean是一款功能强大的Windows系统优化工具&#xff0c;专为普通用户设计&…

作者头像 李华
网站建设 2026/6/7 6:53:07

TouchGal革命性Galgame社区平台:构建属于你的二次元文化净土

重新定义Galgame社区体验 【免费下载链接】kun-touchgal-next TouchGAL是立足于分享快乐的一站式Galgame文化社区, 为Gal爱好者提供一片净土! 项目地址: https://gitcode.com/gh_mirrors/ku/kun-touchgal-next 在当今数字化时代&#xff0c;Galgame爱好者迫切需要一片真…

作者头像 李华
网站建设 2026/6/8 8:38:19

OmenSuperHub:暗影精灵笔记本完全离线性能控制解决方案

还在为官方软件的网络连接问题和隐私泄露风险而烦恼吗&#xff1f;OmenSuperHub为您提供了一款专为暗影精灵笔记本设计的完全离线性能控制工具。这款开源软件专注于硬件性能优化和智能散热管理&#xff0c;让您的游戏本在保持最佳状态的同时享受纯净的使用环境。 【免费下载链接…

作者头像 李华