登录
首页 >  文章 >  java教程

Java生成唯一UUID方法详解

时间:2026-01-19 09:06:41 106浏览 收藏

积累知识,胜过积蓄金银!毕竟在文章开发的过程中,会遇到各种各样的问题,往往都是一些细节知识点还没有掌握好而导致的,因此基础知识点的积累是很重要的。下面本文《Java如何生成唯一UUID标识符》,就带大家讲解一下知识点,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~

UUID.randomUUID()是最常用且稳妥的UUID生成方式,基于SecureRandom生成v4版本,碰撞概率极低,适用于订单号等场景,但不保证有序、不含时间戳,不宜直接用作MySQL主键。

在Java中如何使用UUID生成唯一标识_Java唯一ID工具说明

UUID.randomUUID() 是最常用也最稳妥的生成方式

Java 内置的 UUID 类提供了开箱即用的全局唯一标识生成能力,无需引入第三方依赖。调用 UUID.randomUUID() 会基于随机数(伪随机,使用 SecureRandom)生成一个 version 4 的 UUID,碰撞概率极低(约 2^122 分之一),适合绝大多数业务场景——比如订单号、消息 ID、临时 token 等。

注意它不保证单调递增或时间有序,也不含时间戳信息,纯靠随机性保障唯一性。

UUID id = UUID.randomUUID();
String idStr = id.toString(); // 如 "f47ac10b-58cc-4372-a567-0e02b2c3d479"
  • toString() 返回带连字符的标准格式(32 字符 + 4 个 -),共 36 字符;如需无分隔符形式,可用 id.toString().replace("-", "")
  • 不要用 UUID.nameUUIDFromBytes(byte[]) 除非你明确需要“相同输入必得相同输出”的确定性哈希行为,否则容易因字节序列构造不当导致重复
  • 避免在高并发循环中频繁调用 randomUUID() ——虽然线程安全,但底层 SecureRandom 在某些 JDK 版本(尤其旧版 Linux 系统)可能因熵池耗尽而阻塞

别直接用 UUID 作数据库主键(尤其 MySQL InnoDB)

UUID 字符串作为主键存在明显性能隐患:长度大(36 字符)、无序写入导致 B+ 树频繁分裂和页碎片,插入吞吐下降显著。MySQL 官方文档明确建议避免用 UUID 做聚簇索引键。

如果必须用 UUID,推荐两种折中方案:

  • UUID 存为 BINARY(16):先用 UUID.getMostSignificantBits()getLeastSignificantBits() 拆成两个 long,再组合为 16 字节数组,可节省空间并提升索引效率
  • 用 UUID 做逻辑 ID,另设自增 BIGINT 主键:业务层用 UUID 标识资源,数据库用整型主键维护关系和性能

PostgreSQL 对 UUID 支持更好(原生 UUID 类型 + 高效索引),但依然不建议直接用其做高频写入表的主键。

需要时间有序?考虑 Snowflake 或 ULID,而非 UUID

UUID v4 完全随机,无法反映生成时间或支持按 ID 排序查询。若你需要“新 ID 总是比旧 ID 大”“能从 ID 反推大致生成时间”或“兼容分布式且无中心节点”,UUID 不是合适选择。

Java 生态中有成熟替代:

  • Twitter Snowflake 衍生实现(如 twitter-snowflakejava-snowflake):64 位整型,含时间戳、机器 ID、序列号
  • ULID(Universally Unique Lexicographic Identifier):字符串格式,128 位,前 48 位为毫秒级时间戳,天然有序,Java 有 ulid-creator

例如用 ulid-creator

import com.github.f4b6a3.ulid.Ulid;
Ulid ulid = Ulid.random(); // 返回类似 "01ARZ3NDEKTSV4RRFFQZ8SBSNY" 的字符串
String idStr = ulid.toString();

它比 UUID 更短(26 字符)、可排序、含时间信息,且仍保持全局唯一性。

测试时慎用固定 UUID,避免污染断言逻辑

单元测试中常有人写 UUID.fromString("00000000-0000-0000-0000-000000000000") 作占位 ID,这看似方便,实则埋下隐患:

  • 多个测试用同一固定值,可能导致缓存冲突、数据库唯一约束失败
  • 一旦业务代码误将该“零值”当作特殊语义(如“未设置”),测试会掩盖真实逻辑缺陷
  • 更安全的做法是每次测试生成新 UUID.randomUUID(),或用 Mockito 模拟 UUID 工厂返回可控实例

真正需要可重现 ID 的集成测试场景,建议封装一个测试专用的 TestUuidProvider,内部维护计数器生成确定性但非零的 UUID,而非硬编码字符串。

到这里,我们也就讲完了《Java生成唯一UUID方法详解》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>