4

好吧,我一直在寻找在 java 代码中生成 UID 的方法(其中大多数也进入了 stackoverflow)。最好是使用 java 的 UUID 来创建唯一的 id,因为它使用时间戳。但我的问题是它是 128 位长,我需要一个较短的字符串,比如 14 或 15 个字符。所以,我设计了以下代码来做到这一点。

Date date = new Date();
Long luid = (Long) date.getTime();
String suid = luid.toString();
System.out.println(suid+": "+suid.length() + " characters");

Random rn = new Random();
Integer long1 = rn.nextInt(9);
Integer long2 = rn.nextInt(13);

String newstr = suid.substring(0, long2) + " " + long1 + " " + suid.subtring(long2);
System.out.println("New string in spaced format: "+newstr);
System.out.println("New string in proper format: "+newstr.replaceAll(" ", ""));

请注意,我只是显示间隔格式和格式正确的字符串,仅用于与原始字符串进行比较。

这会保证每次都有 100% 的唯一 ID 吗?或者你认为这些数字有可能重复吗?此外,我可以在开始或结束时将随机数插入“可能”创建重复数字的随机位置,而不是插入。这是为了完成我的 UID 所需的长度。尽管如果您需要少于 13 个字符的 UID,这可能不起作用。

有什么想法吗?

4

2 回答 2

3

如果这是一个分布式系统,这当然是行不通的,但是像下面这样的东西怎么样。

private AtomicLong uniqueId = new AtomicLong(0);
...
// get a unique current-time-millis value 
long now;
long prev;
do {
    prev = uniqueId.get();
    now = System.currentTimeMillis();
    // make sure now is moving ahead and unique
    if (now <= prev) {
        now = prev + 1;
    }
    // loop if someone else has updated the id
} while (!uniqueId.compareAndSet(prev, now));

// shuffle it
long result = shuffleBits(now);
System.out.println("Result is " + Long.toHexString(result));

public static long shuffleBits(long val) {
    long result = 0;
    result |= (val & 0xFF00000000000000L) >> 56;
    result |= (val & 0x00FF000000000000L) >> 40;
    result |= (val & 0x0000FF0000000000L) >> 24;
    result |= (val & 0x000000FF00000000L) >> 8;
    result |= (val & 0x00000000FF000000L) << 8;
    result |= (val & 0x0000000000FF0000L) << 24;
    result |= (val & 0x000000000000FF00L) << 40;
    result |= (val & 0x00000000000000FFL) << 56;
    return result;
}

可以改进位混洗以在每次迭代中产生更多的值变化。您提到您不希望数字是连续的,但您没有指定完全随机性的要求。

当然不如数据库操作,UUID但比数据库操作快。

于 2011-11-16T02:40:45.987 回答
1

简单的方法是使用可用的数据库序列。如果不是,您可以按如下方式模拟它们:

  1. 创建一个表,其中有一列将保存迄今为止使用的最大值(最初为 0)。一些应用程序会创建多行,其中每一行控制一个特定的唯一 ID,但您真正需要的只是一行。对于此示例,假设表结构如下:

    ID_TABLE
    ID_NAME    VARCHAR(40); -- Or whatever type is appropriate
    ID_COLUMN  INTEGER; -- Or whatever type is appropriate
    
  2. 每个进程通过执行以下操作来保留行:

    a. Begin Txn;
    b. Update ID_TABLE set ID_VALUE = ID_VALUE + <n> where ID_NAME = <name>;     
    c. Select ID_VALUE from ID_TABLE where ID_NAME = <name>;  
    d. Commit Txn;
    

    如果这一切都成功完成,那么您刚刚保留了从 (val - n + 1) 到 val 的范围,其中 val 是步骤 c 的返回值。以上。

  3. 每个进程从它保留的范围内分发 ID。如果进程是多线程的,它必须提供同步以确保每个值最多分发一次。当它耗尽其价值供应时,它会返回第 2 步并保留更多价值。请注意,并非所有保留的值都可以保证使用。如果一个进程在没有使用它保留的所有值的情况下终止,则未使用的值将丢失并且永远不会使用。

于 2011-11-13T03:36:51.793 回答