UUID.randomUUID() 是最常用但非最优的UUID生成方式,底层用SecureRandom生成version 4随机UUID,高并发下有锁竞争、存储开销大、不可排序;适合低频单机场景,数据库主键应存为byte[16];需确定性ID时用nameUUIDFromBytes();解析字符串须校验格式;UUID不等同于Snowflake类分布式ID。
Java 的 UUID 类本身不生成 ID,它只是对 128 位值的封装和解析工具;真正生成随机 UUID 的是静态方法 UUID.randomUUID()。这个方法底层调用 SecureRandom(非伪随机),每次调用都产生一个 version 4(随机生成)的 UUID,形如 "f47ac10b-58cc-4372-a567-0e02b2c3d479"。
但它不是“唯一 ID 生成器”的万能解:高并发下频繁调用 UUID.randomUUID() 可能因 SecureRandom 内部锁竞争导致延迟抖动;生成的字符串长度固定 36 字符(含连字符),存储和索引成本高于 16 字节二进制或紧凑编码;且无时间序、不可排序,不利于数据库分页或范围查询。
UUID.randomUUID() 足够直接byte[16] 存储(去掉连字符后 hex 解码),避免字符串索引膨胀SecureRandom 实例并手动构造 version 4 UUID,减少锁开销当你需要“相同输入永远得到相同 UUID”,比如把用户邮箱映射为稳定 ID,或为 API 路径生成缓存 key,应使用 UUID.nameUUIDFromBytes(byte[])。它基于 MD5 哈希(RFC 4122 定义),生成的是 version 3 UUID,结果完全确定、可重现。
注意:输入字节数组不能为 null,且推荐先 UTF-8 编码再传入,否则中文等字符会出错:
String input = "user@example.com"; UUIDuuid = UUID.nameUUIDFromBytes(input.getBytes(StandardCharsets.UTF_8));
randomUUID() 不同,它不依赖系统熵源,无性能瓶颈
调用 UUID.fromString(String) 解析外部传入的 UUID 字符串(如 HTTP Header、JSON 字段)前,务必确认格式合法。非法输入如缺少连字符、含非十六进制字符、长度不对,都会直接触发 IllegalArgumentException,而非返回 null。
"abc123"、"f47ac10b-58cc-4372-a567-0e02b2c3d47"(少一位)、"f47ac10b-58cc-4372-a567-0e02b2c3d47g"(含 g)^[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}$
UUID.fromString() 解析自定义短 ID(如 Snowflake 字符串),它只认标准 UUID 格式UUID 是全局唯一标识符(Globally Unique Identifier),强调“冲突概率极低”,但不保证严格单调、不保证时间局部性、不提供序列号语义。而像 Twitter Snowflake、百度 UidGenerator、美团 Leaf 这类方案,核心目标是:毫秒级时间戳 + 机器 ID + 序列号,生成 64 位 long 型递增 ID,便于数据库索引、分库分表路由、监控追踪。
UUID 类不支持生成 version 1)真正容易被忽略的是:很多人把 UUID.randomUUID().toString() 当成“分布式唯一 ID”直接塞进订单表主键,却没意识到它让 MySQL 的聚簇索引彻底失效——随机字符串导致大量页分裂和磁盘随机写,压测时 QPS 断崖下跌。