我需要在Java中生成一个唯一的10位数ID。以下是此ID的限制:
我到目前为止找到的最佳解决方案如下:
private static int inc = 0;
private static long getId(){
long id = Long.parseLong(String.valueOf(System.currentTimeMillis())
.substring(1,10)
.concat(String.valueOf(inc)));
inc = (inc+1)%10;
return id;
}
此解决方案存在以下问题:
创建此ID的其他任何解决方案?
我没有想到的任何其他问题?
感谢您的帮助,
答案 0 :(得分:8)
这是对你的一个小小的改进,但应该是有弹性的。
private static final long LIMIT = 10000000000L;
private static long last = 0;
public static long getID() {
// 10 digits.
long id = System.currentTimeMillis() % LIMIT;
if ( id <= last ) {
id = (last + 1) % LIMIT;
}
return last = id;
}
因为它应该以相对较短的循环速率管理高达每秒1000次。要延长周期费率(但缩短分辨率),您可以使用(System.currentTimeMillis() / 10) % 10000000000L
或(System.currentTimeMillis() / 100) % 10000000000L
。
答案 1 :(得分:2)
这可能是一个疯狂的想法,但它的想法:)。
java.util.UUID.randomUUID().toString()
第二次将生成的字符串转换为字节数组(byte[]
)
然后将其转换为长缓冲区:java.nio.ByteBuffer.wrap( byte
digest[] ).asLongBuffer().get()
截断为10位
不确定该方法的唯一性,我知道你可以依赖UUID的唯一性,但是没有检查它们的唯一性转换和截断为10位长的数字。
示例来自JavaRanch,也许还有更多。
编辑:由于您只能使用10位数字,因此简单的随机生成器就足够了,请查看SO上的问题/答案:Java: random long number in 0 <= x < n range
答案 2 :(得分:0)
什么意味着它必须是独一无二的?甚至跨越更多当前运行的实例?它打破了你的实施。
如果它必须在整个Universe中是唯一的,那么最好的解决方案是使用UUID,因为它是经过数学验证的标识符生成器,因为它为每个Universe生成唯一值。不太准确的数字会带来碰撞。
当只有一个并发实例时,您可以以毫秒为单位获取当前时间并使用增量解决10ms问题。如果您在数字中牺牲了适当数量的最后位置,您可以在一毫秒内获得许多数字。我想定义精度 - 我的意思是你每秒需要多少唯一数字。您将使用此方法在没有任何持久性的情况下解决问题。
答案 3 :(得分:0)
private static AtomicReference currentTime = new AtomicReference&lt;&gt;(System.currentTimeMillis());
public static Long nextId() {
return currentTime.accumulateAndGet(System.currentTimeMillis(), (prev, next) -> next > prev ? next : prev + 1) % 10000000000L;
}