【发布时间】:2017-01-22 23:48:33
【问题描述】:
我必须在 Java 中生成一个唯一编号(对于运行代码的我的机器),以便在 C++ 中它可以对应于uint32_t。一般来说,其他 C++ 程序应该能够正确地读取这个唯一编号为uint32_t。我在一个字节数组中发送这个信息,其他 C++ 程序正在反序列化它以获得这个数字。到目前为止,我无法在 C++ 程序中更改此数字的数据类型。
private static final AtomicInteger clientId = new AtomicInteger(0);
// will be called many time when the program is running after initialization
public static int getClientId() {
int counter = clientId.incrementAndGet();
return counter == Integer.MAX_VALUE ? 0 : counter;
}
// called only once at startup
public static void setClientId(final int counter) {
clientId.addAndGet(counter == Integer.MAX_VALUE ? 0 : counter);
}
所以我想出了上面的代码。我的setClientId 方法在应用程序启动(初始化阶段)期间只会被调用一次:
- 如果这台机器是第一次运行代码,就像是一台新机器,那么我会将
0传递给setClientId方法。 - 如果这台机器已经在运行代码但我们重新启动了服务器,那么我将在数据库中查找这台机器最后保存的值是什么(因为我们也每隔几分钟将这个值保存在 db 中)然后我将该值传递给
setClientId方法。
然后当我的程序运行时(在初始化之后),它会继续调用getClientId 方法来给我实际唯一的clientId。
这是为我的机器生成唯一 ID 的正确方法,对于 C++ 程序来说,这将是 uint32_t?我还确保如果该值达到Integer.MAX_VALUE,则将其设置为 0 并重新开始。我必须这样做吗?或者我也可以为此 uint32_t 取负值?
基本上,我想为我的机器生成一个唯一编号,该编号应始终对应于uint32_t。如果机器是新机器,我们可以从数字 0 开始(bcoz 当我们在数据库中查找这台机器的值时,不会有任何值,所以我们将从 0 开始)但是如果机器之前已经运行过这段代码,然后从数据库中最后保存的值开始。
【问题讨论】:
-
Java 整数是有符号的,如果你想要完整的范围,你也必须取负值。
-
那么
uint32_t也能理解负整数值吗?请原谅我在这里对 C++ 的无知。 -
只有当您期望生成超过 2,147,483,647 个唯一 ID 时才会出现问题。但是,如果您将 Java 签名的
int(可能为负数)序列化为 4 个字节,并将 C++ 程序中的这 4 个字节反序列化为uint32_t,C++ 程序会将负 Java 整数视为 2147483648 到4294967295. -
我明白了。然后最好只发送正整数值,以便它只能反序列化正整数值。对于长时间运行代码的机器,它有可能超过
2,147,483,647 unique ID's,一旦达到该限制,我就会再次将其重置回0。 -
@david 好主意。另一种方法是在 Java 中使用
long并手动将内容转换为 C 期望的格式(仅发送低 32 位)。
标签: java c++ multithreading thread-safety atomicinteger