Лучший способ преобразовать целое число со знаком в длинное без знака?
для некоторых хэш-функций в Java было бы неплохо увидеть значение как целое число без знака (например, для сравнения с другими реализациями), но Java поддерживает только подписанные типы. Мы можем преобразовать подписанный int
на "unsigned"long
такие как:
public static final int BITS_PER_BYTE = 8;
public static long getUnsignedInt(int x) {
ByteBuffer buf = ByteBuffer.allocate(Long.SIZE / BITS_PER_BYTE);
buf.putInt(Integer.SIZE / BITS_PER_BYTE, x);
return buf.getLong(0);
}
getUnsignedInt(-1); // => 4294967295
однако это решение кажется излишним для того, что мы действительно делаем. Есть ли более эффективный способ достичь того же самого?
7 ответов
что-то вроде этого?
int x = -1;
long y = x & 0x00000000ffffffffL;
или я что-то пропустила?
public static long getUnsignedInt(int x) {
return x & 0x00000000ffffffffL;
}
стандартным способом в Java 8 является Integer.toUnsignedLong(someInt)
, что эквивалентно @Mysticial это!--3-->.
гуавы обеспечивает UnsignedInts.toLong(int)
...а также множество других утилит для целых чисел без знака.
вы можете использовать такую функцию, как
public static long getUnsignedInt(int x) {
return x & (-1L >>> 32);
}
однако в большинстве случаев вам не нужно этого делать. Можно использовать обходные пути. например,
public static boolean unsignedEquals(int a, int b) {
return a == b;
}
дополнительные примеры обходных путей для использования значений без знака. unsigned utility class
другое решение.
public static long getUnsignedInt(int x) {
if(x > 0) return x;
long res = (long)(Math.pow(2, 32)) + x;
return res;
}
просто мои 2 цента здесь, но я думаю, что это хорошая практика, чтобы использовать:
public static long getUnsignedInt(int x) {
return x & (~0L); // ~ has precedence over & so no real need for brackets
}
вместо:
возвращает X & 0xFFFFFFFFL;
в этой ситуации вас не волнует, сколько " F " имеет маска. Это всегда будет работать!