UUID做主键,优点,缺点!

本文详细介绍了UUID的概念及其生成原理,探讨了UUID作为唯一标识符的优势与不足,并提供了一个实用的UUID生成函数示例。
那么我们先看看什么是UUID?简单的说,UUID是指在一台机器上生成的数字,它保证对在同一时空中的所有机器都是唯一的。在UUID的算法中,可能会用到诸如网卡MAC地址,IP,主机名,进程ID等信息以保证其独立性.
优点:
能够保证独立性,程序可以在不同的数据库间迁移,效果不受影响。
保证生成的ID不仅是表独立的,而且是库独立的,这点在你想切分数据库的时候尤为重要。
缺点:
比较占地方,和INT类型相比,存储一个UUID要花费更多的空间。
使用UUID后,URL显得冗长,不够友好。

那么到底使用不使用UUID做主键呢?? http://hi.baidu.com/thinkinginlamp/blog/item/c609d10979710e81d0581b60.html

那么如何生成UUID,一个自动生成不重复文本的函数,Hibernate提供的uuid.
hibernate中的uuid是自动生成主键的一种机制,但是它依赖于hiberbate包,刚刚将它从Hibernate中提取出来,(从其他博客和公司的组件代码中发现了UUID的生成类)供各位使用.

import java.net.InetAddress;

/**
*
* @author zhangguang
*
*/
public class UUIDHexGenerator {

private static final int IP;
private static UUIDHexGenerator instance = null;
private static short counter = (short) 0;
private static final int JVM = (int) ( System.currentTimeMillis() >>> 8 );
private String sep = "";
static {
int ipadd;
try {
ipadd = toInt( InetAddress.getLocalHost().getAddress() );
}
catch (Exception e) {
ipadd = 0;
}
IP = ipadd;
}
protected String format(int intval) {
String formatted = Integer.toHexString(intval);
StringBuffer buf = new StringBuffer("00000000");
buf.replace( 8-formatted.length(), 8, formatted );
return buf.toString();
}

protected String format(short shortval) {
String formatted = Integer.toHexString(shortval);
StringBuffer buf = new StringBuffer("0000");
buf.replace( 4-formatted.length(), 4, formatted );
return buf.toString();
}

public static int toInt( byte[] bytes ) {
int result = 0;
for (int i=0; i<4; i++) {
result = ( result << 8 ) - Byte.MIN_VALUE + (int) bytes[i];
}
return result;
}
protected int getJVM() {
return JVM;
}


protected short getCount() {
synchronized(UUIDHexGenerator.class) {
if (counter<0) counter=0;
return counter++;
}
}
/**
* @author zhangguang
*
* @return UUID. 一組 UUID,係由一串 16 位元組(亦稱 16 字節,或 128 位元)的16進位數字所構成
*/
public String generate() {
return new StringBuffer(36)
.append( format( getIP() ) ).append(sep)
.append( format( getJVM() ) ).append(sep)
.append( format( getHiTime() ) ).append(sep)
.append( format( getLoTime() ) ).append(sep)
.append( format( getCount() ) )
.toString();
}


protected int getIP() {
return IP;
}

protected short getHiTime() {
return (short) ( System.currentTimeMillis() >>> 32 );
}
protected int getLoTime() {
return (int) System.currentTimeMillis();
}

public static UUIDHexGenerator getInstance()
{
if (instance == null)
instance = new UUIDHexGenerator();
return instance;
}




public static void main( String[] args ) throws Exception {

for ( int i=0; i<10; i++) {
String id = getInstance().generate();
System.out.println(id);
String id2 = getInstance().generate();
System.out.println(id2);
}

}



}
<think>嗯,用户想了解Java中使用UUID作为ID生成方式的优缺点。首先,我得回忆一下之前提到的引用内容。引用1提到Java生成UUID的方法,并说明其目的是为了避免分布式系统中的ID重复。引用2和4详细对比了自增ID、雪花ID和UUID的优缺点,指出UUID的全局唯一性和存储空间大的问题。引用3解释了UUID的结构和核心特性。 用户可能是在设计数据库时需要选择主生成方式,所以需要权衡不同方案的利弊。他们可能关心分布式系统中的唯一性、性能影响以及存储成本。根据引用内容,UUID优点包括全局唯一、无需中央协调、生成简单,而缺点有存储空间大、查询效率低、隐私泄露风险。 需要组织这些信息,分点列出优缺点,可能还要提到有序UUID作为优化方法。同时,用户可能不知道有序UUID的存在,所以应该补充这一点。还要注意使用正确的LaTeX格式,比如在提到128位时用$2^{128}$。最后,生成的相关问题要围绕实际应用,比如性能优化、适用场景、隐私问题等。</think>### Java UUID作为ID生成方式的优缺点分析 #### **优点分析** 1. **全局唯一性** UUID基于$2^{128}$的大空间随机生成,理论上在分布式系统中可实现无重复标识符,无需中央协调即可保证全网唯一[^2][^3]。例如,在分库分表场景下,不同节点生成的UUID不会冲突[^1]。 2. **无中心化依赖** 生成过程完全本地化,无需依赖数据库自增序列或网络请求,适用于高并发场景[^2][^3]。 3. **安全性** 随机生成的UUID无递增规律,可避免被恶意猜测主值(但需注意部分版本可能包含MAC地址,存在隐私泄露风险)[^4]。 4. **简单易用** Java原生支持`UUID.randomUUID()`方法,生成代码简洁,如: ```java String uuid = UUID.randomUUID().toString().replace("-", ""); ``` --- #### **缺点分析** 1. **存储成本高** UUID通常以36字符字符串形式存储(如`550e8400-e29b-41d4-a716-446655440000`),占用`varchar(36)`及以上空间,远大于自增ID(4字节整数)或雪花算法ID(8字节长整数)[^2][^4]。 2. **查询性能问题** 无序的UUID会导致数据库索引页分裂,增加磁盘随机读取开销,尤其是在批量查询时性能显著下降。引用[4]指出,可通过**有序UUID**(如Hibernate的`UUIDHexGenerator`)优化此问题[^2][^4]。 3. **可读性差** 相比自增ID,UUID无法直观反映数据插入顺序,调试和维护成本较高。 4. **隐私风险** UUID版本1基于MAC地址和时间戳生成,可能泄露设备信息。推荐使用版本4(完全随机)或版本5(基于命名空间哈希)[^3]。 --- #### **适用场景建议** - **分布式系统**:需跨节点唯一标识时优先选择[^2]。 - **临时数据**:如会话ID、文件上传标识。 - **安全敏感场景**:需隐藏数据规律时(配合有序UUID优化性能)。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值