Когда я построил маску для получения наиболее значимого бита в формате дополнения 2, я обнаружил неожиданное поведение.
Чтобы проверить, активен ли самый старший бит в 8-битном знаке число, я мог бы получить бит следующим образом.
byte value = -1;
long byteSignMask = 0b1000_0000;
value & byteSignMask;
Результат идентичен независимо от того, что я использую 0b1000_0000
или 1L << 7
для byteSignMask
. На самом деле проходит следующий код.
long byteSign1 = 1L << 7;
long byteSign2 = 0b1000_0000;
// OK
assertEquals(byteSign1, byteSign2);
Но я сделал это для типа int; аналогично, результат ожидался.
long intSign1 = 1L << 31;
long intSign2 = 0b1000_0000_0000_0000_0000_0000_0000_0000;
// Fail: expected:<2147483648> but was:<-2147483648>
assertEquals(intSign1, intSign2);
На самом деле, они разные.
// intSign1 = 10000000000000000000000000000000
System.out.println("intSign1 = " + Long.toBinaryString(intSign1));
// intSign2 = 1111111111111111111111111111111110000000000000000000000000000000
System.out.println("intSign2 = " + Long.toBinaryString(intSign2));
Похоже, буквальная маска целого числа (intSign1
) дополнена слева 1, в то время как операция сдвига не вызывает такого эффекта.
Почему целое число, выраженное двоичным литералом, автоматически дополняется слева единицей? Есть ли официальная документация, описывающая такое поведение?