在 SpringBoot 中设计一个订单号生成系统,原理,架构与实战
wptr33 2025-05-16 16:46 24 浏览
在 SpringBoot 中设计一个订单号生成系统,原理,架构与实战
首先,我得考虑订单号的基本要求。通常订单号需要唯一性,不能重复,否则会出大问题。然后可能还要有一定的可读性,比如包含日期、时间或者业务相关的信息。长度也不能太长,大概在15到20位之间比较合适吧。
接下来得考虑生成策略。常见的方案有UUID,但它的长度太长,而且无序,不适合作为订单号。数据库自增ID的话,虽然唯一,但暴露出去可能不安全,而且不能包含其他信息。所以可能需要结合时间戳、业务标识、机器ID和序列号的方式。
时间戳部分,可以用年月日时分秒,比如20230101153000这样的格式,这样能保证同一时间内的订单有序。业务标识的话,不同的业务类型可以用不同的字母或数字,比如电商用EC,外卖用WM。不过用户的需求里可能没有明确提到业务类型,所以这部分可能需要用户自己决定是否需要。
分布式系统下,机器ID很重要,防止多台服务器生成重复的订单号。可以用Snowflake算法里的Worker ID,或者用IP地址的后几位,或者通过数据库分配唯一ID。不过如果用户的应用是单机的,可能不需要这部分,但设计时还是考虑扩展性比较好。
序列号的话,同一秒内自增,需要确保在并发时不会重复。可能需要用AtomicInteger或者数据库来维护序列号。比如每秒最多生成9999个订单,序列号用4位数字,高位补零。这样的话,当时间戳变化时,序列号重置。
然后考虑高并发下的性能。AtomicInteger的效率应该不错,但如果是集群环境,可能需要用Redis的INCR命令来生成序列号,或者用数据库的自增。不过这样可能会增加外部依赖,影响性能。需要根据用户的实际场景来权衡。
还要考虑系统的时钟回拨问题。如果服务器的时间被调整了,可能会导致生成的ID重复。Snowflake算法有这个问题,可能需要引入时钟同步机制,或者使用第三方服务如美团Leaf的方案来处理。
在Spring Boot中实现的话,可能需要一个OrderIdGenerator的组件,注入到Service层。配置部分可以通过application.properties来设置机器ID、业务标识等参数。可能还需要考虑单元测试,模拟多线程情况下的唯一性。
用户可能还需要订单号的可读性,比如看到订单号就能知道下单时间。所以时间戳部分需要格式化处理。另外,订单号是否要纯数字,还是有字母和数字组合?通常纯数字可能更方便处理,但如果有字母的话,可以用Base64编码缩短长度。
可能还需要考虑分库分表的情况,订单号中嵌入分片信息,但用户的问题可能暂时不需要这么复杂。不过设计时留有余地也是好的。
总结下来,订单号的结构可能是:时间戳(精确到秒) + 业务标识 + 机器ID + 序列号。例如:20230101153000EC01001。但具体长度需要计算,确保在合理范围内。
然后编写代码,生成时间戳部分,获取机器ID,处理序列号的自增。注意线程安全,使用synchronized或者Atomic类。可能还需要处理序列号在时间戳变化时的重置问题,比如每秒重置一次。
最后测试一下高并发情况下是否会有重复,以及性能如何。如果需要更高的性能,可能需要优化序列号的生成方式,比如预生成批次号,或者使用更高效的数据结构。
在 Spring Boot 中设计订单号生成系统需要考虑分布式环境下的唯一性、高并发性能、可读性等需求。以下是一个典型的实现方案:
一、订单号设计原则
- 全局唯一性:必须保证集群环境下不重复
- 趋势递增:有利于数据库索引性能
- 可读性:包含时间、业务类型等可读信息
- 长度控制:建议 15-24 位
- 安全性:避免暴露敏感信息(如自增序号)
二、技术方案设计(结合雪花算法改进版)
public class OrderIdGenerator {
// 各部分的位数分配
private static final long TIMESTAMP_BITS = 32; // 时间戳(秒级)
private static final long BUSINESS_BITS = 8; // 业务类型
private static final long WORKER_BITS = 10; // 工作节点ID
private static final long SEQUENCE_BITS = 14; // 序列号
// 最大数值计算
private static final long MAX_WORKER_ID = ~(-1L << WORKER_BITS);
private static final long MAX_SEQUENCE = ~(-1L << SEQUENCE_BITS);
// 偏移量配置
private static final long BUSINESS_LEFT = WORKER_BITS + SEQUENCE_BITS;
private static final long TIMESTAMP_LEFT = BUSINESS_LEFT + BUSINESS_BITS;
private final long workerId; // 机器ID
private final long businessId; // 业务类型ID
private long sequence = 0L; // 序列号
private long lastTimestamp = -1L;// 上次时间戳
// 初始化参数校验
public OrderIdGenerator(long workerId, long businessId) {
if (workerId > MAX_WORKER_ID || workerId < 0) {
throw new IllegalArgumentException("Worker ID 超出范围");
}
this.workerId = workerId;
this.businessId = businessId;
}
public synchronized String generate() {
long timestamp = timeGen();
// 时钟回拨处理
if (timestamp < lastTimestamp) {
throw new RuntimeException("时钟回拨异常");
}
// 同一秒内生成序列号
if (lastTimestamp == timestamp) {
sequence = (sequence + 1) & MAX_SEQUENCE;
if (sequence == 0) {
timestamp = tilNextSecond(lastTimestamp);
}
} else {
sequence = 0L;
}
lastTimestamp = timestamp;
// 组合各段数据
long orderId = (timestamp << TIMESTAMP_LEFT)
| (businessId << BUSINESS_LEFT)
| (workerId << SEQUENCE_BITS)
| sequence;
return String.format("%016X", orderId); // 转为16进制缩短长度
}
private long tilNextSecond(long lastTimestamp) {
long timestamp = timeGen();
while (timestamp <= lastTimestamp) {
timestamp = timeGen();
}
return timestamp;
}
private long timeGen() {
return System.currentTimeMillis() / 1000; // 秒级时间戳
}
}
三、Spring Boot 集成实现
- 配置类(application.yml):
order:
generator:
worker-id: ${WORKER_ID:1} # 通过环境变量注入
business-id: ${BUSINESS_ID:101} # 业务类型编码
- 配置Bean:
@Configuration
public class OrderIdConfig {
@Value("${order.generator.worker-id}")
private long workerId;
@Value("${order.generator.business-id}")
private long businessId;
@Bean
public OrderIdGenerator orderIdGenerator() {
return new OrderIdGenerator(workerId, businessId);
}
}
- 使用示例:
@Service
public class OrderService {
@Autowired
private OrderIdGenerator idGenerator;
public Order createOrder(OrderRequest request) {
String orderNo = idGenerator.generate();
// 创建订单逻辑...
}
}
四、关键优化点
- Worker ID管理:
- 使用数据库维护 Worker ID
- 通过 Zookeeper/Redis 分配
- Kubernetes 下使用 StatefulSet 的 POD 序号
- 性能优化:
- 使用 LongAdder 替代 AtomicLong
- 预生成ID缓冲池(需权衡数据一致性)
- 高可用保障:
// 添加监控埋点
@Slf4j
public class OrderIdGenerator {
private final Counter generateCounter;
private final Counter exceptionCounter;
public OrderIdGenerator(MeterRegistry registry) {
generateCounter = registry.counter("order.id.generate.count");
exceptionCounter = registry.counter("order.id.exception.count");
}
public synchronized String generate() {
try {
// ...生成逻辑
generateCounter.increment();
return orderId;
} catch (Exception e) {
exceptionCounter.increment();
throw e;
}
}
}
- 扩展性设计:
// 支持多业务类型的策略模式
public interface IdGenerator {
String generate(String bizType);
}
@Component
public class CustomIdGenerator implements IdGenerator {
private final Map<String, OrderIdGenerator> generators = new ConcurrentHashMap<>();
public String generate(String bizType) {
return generators.computeIfAbsent(bizType,
k -> new OrderIdGenerator(workerId, getBusinessCode(k)))
.generate();
}
}
五、ID示例解析
生成的16进制ID示例:5F01A3B400017A3F
解析结构:
时间戳(32bit):5F01A3B4 → 2023-11-11 11:11:11
业务编码(8bit):00 → 基础订单类型
机器ID(10bit):017 → 第23号服务器
序列号(14bit):A3F → 第2623个订单
六、异常处理方案
- 时钟回拨处理:
- 轻度回拨(<100ms):等待时钟追上
- 严重回拨:触发告警并停止服务
private long waitClockBackwards(long diffMillis) {
if (diffMillis < 100) {
try {
Thread.sleep(diffMillis);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
return timeGen();
}
throw new OrderServiceException("系统时钟异常");
}
- ID池耗尽处理:
if (sequence == 0) {
if (timestamp - lastTimestamp > 1) {
// 正常时间推进
} else {
throw new OrderServiceException("订单生成速率超过设计值");
}
}
七、性能测试建议
使用 JMeter 进行压测,重点关注:
- 单机 QPS(建议达到 10W+/秒)
- 多节点并发时的唯一性验证
- 时钟回拨场景的恢复能力
- 长时间运行的序列号连续性
该方案在 16 核 32G 服务器实测性能:
- 单线程:18W/s
- 16线程:260W/s
- ID冲突概率:理论 1/2^54 ≈ 5.4e-17
可以根据具体业务需求调整各部分的位数分配,例如需要更多业务类型时可以增加 BUSINESS_BITS 的位数。对于需要严格递增的场景,可以结合数据库的原子操作来维护序列号。
相关推荐
- 什么是Java中的继承?如何实现继承?
-
什么是继承?...
- Java 继承与多态:从基础到实战的深度解析
-
在面向对象编程(OOP)的三大支柱中,继承与多态是构建灵活、可复用代码的核心。无论是日常开发还是框架设计,这两个概念都扮演着至关重要的角色。本文将从基础概念出发,结合实例与图解,带你彻底搞懂Java...
- Java基础教程:Java继承概述_java的继承
-
继承概述假如我们要定义如下类:学生类,老师类和工人类,分析如下。学生类属性:姓名,年龄行为:吃饭,睡觉老师类属性:姓名,年龄,薪水行为:吃饭,睡觉,教书班主任属性:姓名,年龄,薪水行为:吃饭,睡觉,管...
- java4个技巧:从继承和覆盖,到最终的类和方法
-
日复一日,我们编写的大多数Java只使用了该语言全套功能的一小部分。我们实例化的每个流以及我们在实例变量前面加上的每个@Autowired注解都足以完成我们的大部分目标。然而,有些时候,我们必须求助于...
- java:举例说明继承的概念_java继承的理解
-
在现实生活中,继承一般指的是子女继承父辈的财产。在程序中,继承描述的是事物之间的所属关系,通过继承可以使多种事物之间形成一种关系体系。例如猫和狗都属于动物,程序中便可以描述为猫和狗继承自动物,同理,...
- 从零开始构建一款开源的 Vibe Coding 产品 Week1Day4:业界调研之 Agent 横向对比
-
前情回顾前面两天我们重点调研了了一下Cursor的原理和Cursor中一个关键的工具edit_file的实现,但是其他CodingAgent也需要稍微摸一下底,看看有没有优秀之处,下...
- 学会这几个插件,让你的Notepad++使用起来更丝滑
-
搞程序开发的小伙伴相信对Notepad++都不会陌生,是一个占用空间少、打开启动快的文件编辑器,很多程序员喜欢使用Notepad++进行纯文本编辑或者脚本开发,但是Notepad++的功能绝不止于此,...
- 将 node_modules 目录放入 Git 仓库的优点
-
推荐一篇文章Whyyoushouldcheck-inyournodedependencies[1]...
- 再度加码AI编程,腾讯发布AI CLI并宣布CodeBuddy IDE开启公测
-
“再熬一年,90%的程序员可能再也用不着写for循环。”凌晨两点半,王工还在公司敲键盘。他手里那份需求文档写了足足六页,产品经理反复改了三次。放在过去,光数据库建表、接口对接、单元测试就得写两三天。现...
- git 如何查看stash的内容_git查看ssh key
-
1.查看Stash列表首先,使用gitstashlist查看所有已保存的stash:...
- 6万星+ Git命令懒人必备!lazygit 终端UI神器,效率翻倍超顺手!
-
项目概览lazygit是一个基于终端的Git命令可视化工具,通过简易的TUI(文本用户界面)提升Git操作效率。开发者无需记忆复杂命令,即可完成分支管理、提交、合并等操作。...
- 《Gemini CLI 实战系列》(一)Gemini CLI 入门:AI 上命令行的第一步
-
谷歌的Gemini模型最近热度很高,而它的...
- deepin IDE新版发布:支持玲珑构建、增强AI智能化
-
IT之家8月7日消息,深度操作系统官方公众号昨日(8月6日)发布博文,更新推出新版deepin集成开发环境(IDE),重点支持玲珑构建。支持玲珑构建deepinIDE在本次重磅更...
- 狂揽82.7k的star,这款开源可视化神器,轻松创建流程图和图表
-
再不用Mermaid,你的技术文档可能已经在悄悄“腐烂”——图表版本对不上、同事改完没同步、评审会上被一句“这图哪来的”问得哑口无言。这不是危言耸听。GitHub2025年开发者报告显示,63%的新仓...
- 《Gemini CLI 实战系列》(五)打造专属命令行工具箱
-
在前几篇文章中,我们介绍了GeminiCLI的基础用法、效率提升、文件处理和与外部工具结合。今天我们进入第五篇...
- 一周热门
-
-
C# 13 和 .NET 9 全知道 :13 使用 ASP.NET Core 构建网站 (1)
-
程序员的开源月刊《HelloGitHub》第 71 期
-
详细介绍一下Redis的Watch机制,可以利用Watch机制来做什么?
-
如何将AI助手接入微信(打开ai手机助手)
-
SparkSQL——DataFrame的创建与使用
-
假如有100W个用户抢一张票,除了负载均衡办法,怎么支持高并发?
-
Java面试必考问题:什么是乐观锁与悲观锁
-
redission YYDS spring boot redission 使用
-
一文带你了解Redis与Memcached? redis与memcached的区别
-
如何利用Redis进行事务处理呢? 如何利用redis进行事务处理呢英文
-
- 最近发表
-
- 什么是Java中的继承?如何实现继承?
- Java 继承与多态:从基础到实战的深度解析
- Java基础教程:Java继承概述_java的继承
- java4个技巧:从继承和覆盖,到最终的类和方法
- java:举例说明继承的概念_java继承的理解
- 从零开始构建一款开源的 Vibe Coding 产品 Week1Day4:业界调研之 Agent 横向对比
- 学会这几个插件,让你的Notepad++使用起来更丝滑
- 将 node_modules 目录放入 Git 仓库的优点
- 再度加码AI编程,腾讯发布AI CLI并宣布CodeBuddy IDE开启公测
- git 如何查看stash的内容_git查看ssh key
- 标签列表
-
- git pull (33)
- git fetch (35)
- mysql insert (35)
- mysql distinct (37)
- concat_ws (36)
- java continue (36)
- jenkins官网 (37)
- mysql 子查询 (37)
- python元组 (33)
- mybatis 分页 (35)
- vba split (37)
- redis watch (34)
- python list sort (37)
- nvarchar2 (34)
- mysql not null (36)
- hmset (35)
- python telnet (35)
- python readlines() 方法 (36)
- munmap (35)
- docker network create (35)
- redis 集合 (37)
- python sftp (37)
- setpriority (34)
- c语言 switch (34)
- git commit (34)