Java生成唯一UUID随机ID方法
时间:2026-03-09 22:57:44 475浏览 收藏
Java中生成唯一ID看似简单,实则暗藏诸多陷阱:UUID.randomUUID()虽线程安全却因完全随机导致数据库索引性能差、存储冗余、日志排查不便;盲目用作高频插入场景主键会加剧B+树页分裂;去掉短横线或转为BINARY(16)仅缓解表层问题,无法解决无序性与时间不可追溯的短板;真正需要时间有序、分布式唯一且高效时,应转向Snowflake等时间+机器+序列组合方案;而数据库字段类型错配、测试环境伪随机掩盖真实冲突、业务层缺乏预检机制等问题,更会让“概率唯一”在生产链路中演变为重复事故。选型不是只看生成一行代码,而是通盘考量存储、索引、传输、排查与扩展的全生命周期代价。

UUID.randomUUID() 是最常用也最容易出错的起点
Java 里生成唯一 ID,UUID.randomUUID() 是第一反应,但它生成的是 128 位随机 UUID(version 4),不是递增、不带时间信息、字符串长度固定 36 字符(含 4 个短横线)。很多人直接用它当数据库主键,结果发现索引性能差、存储冗余、排查日志时还得手动去掉短横线。
- 别在高频插入场景(比如订单流水表)直接用
UUID.randomUUID().toString()作主键——B+ 树索引会因随机写导致页分裂严重 - 如果需要“可读性稍好一点”,可以用
UUID.randomUUID().toString().replace("-", "")去掉分隔符,变成 32 位十六进制字符串,但仍是完全随机的 - 注意
UUID是线程安全的,但频繁调用仍有一定对象创建开销;高并发下不如预生成一批缓存再取
想带时间戳又保持唯一?用 Snowflake 或自研时间+随机组合
UUID 本身不提供时间顺序能力。如果你需要“大致按时间排序”“能从 ID 反推生成时间”或“避免数据库索引碎片”,就得换思路。Java 生态里没有内置 Snowflake 实现,得靠第三方库或自己写轻量版。
- 推荐用
TwitterSnowflake的 Java 移植版(如twitter-snowflake或mybatis-plus内置的IdWorker),它生成 long 类型 ID,高位是时间戳(毫秒级)、中位是机器 ID、低位是序列号 - 自己拼接时间戳 + 随机数也能凑合:比如
System.currentTimeMillis() + "-" + ThreadLocalRandom.current().nextInt(1000, 9999),但要注意并发下时间戳重复风险,必须加锁或用AtomicLong做序列保底 - 别用
new Date().getTime()直接当 ID——毫秒级精度在单机高并发下极易重复,且无法区分来源节点
数据库字段类型不匹配会导致隐式转换和查询变慢
生成 ID 后往数据库写,类型选错会让原本的唯一性优势打折扣。MySQL 和 PostgreSQL 对 UUID 的支持差异明显,尤其在索引效率上。
- MySQL 中,用
CHAR(36)存标准 UUID 字符串,但更优解是BINARY(16)——先用UUID.fromString(s).getMostSignificantBits()等方法转成两个 long,再存为字节数组,索引大小直接减半 - PostgreSQL 有原生
UUID类型,支持高效比较和索引,但 JDBC 驱动默认把ResultSet.getObject("id")返回为String,需显式调用getUUID()才能拿到java.util.UUID对象 - 千万别把 UUID 字符串存进
INT或BIGINT字段——会截断或报错Data truncation: Out of range value for column 'id'
测试环境用固定 UUID 会掩盖分布式唯一性问题
单元测试里为了可重现,常写死 UUID.fromString("123e4567-e89b-12d3-a456-426614174000"),这没问题;但集成测试或本地联调时,如果所有服务都用相同“伪随机种子”生成 UUID,就测不出真实冲突场景。
- Spring Boot 测试中,可通过配置类禁用默认
SecureRandom,改用new Random(123)控制随机性,但仅限单 JVM 场景 - 微服务间通过消息传递 ID 时,要确认上游是否真用了
randomUUID(),还是误用了静态 UUID 单例——后者在多实例部署下必然重复 - 线上查重逻辑不能只依赖数据库唯一索引报错,得在业务层加
SELECT COUNT(*) FROM table WHERE id = ?预检,否则异常堆栈里全是SQLIntegrityConstraintViolationException,定位成本高
UUID 的“唯一”是概率意义上的,不是绝对保证。真正麻烦的从来不是生成那一刻,而是 ID 落库、跨系统传输、日志关联、归档检索这一整条链路上,每个环节对格式、长度、排序性、可读性的隐含假设。漏掉任意一环,都可能让“唯一 ID”变成“重复事故源头”。
本篇关于《Java生成唯一UUID随机ID方法》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!
相关阅读
更多>
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
最新阅读
更多>
-
238 收藏
-
111 收藏
-
140 收藏
-
253 收藏
-
188 收藏
-
500 收藏
-
128 收藏
-
491 收藏
-
397 收藏
-
150 收藏
-
318 收藏
-
331 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习