Bitshift a long in Java

2022-09-02 03:12:56

我敢肯定,对于第一个看到它的人来说,这是一个很容易的!

为什么在Java中代码像

   long one = 1 << 0;
   long thirty = 1 << 30;
   long thirtyOne = 1 << 31;
   long thirtyTwo = 1 << 32;

   System.out.println(one+" = "+Long.toBinaryString(1 << 0));
   System.out.println(thirty+" = "+Long.toBinaryString(1 << 30));
   System.out.println(thirtyOne+" = "+Long.toBinaryString(1 << 31));
   System.out.println(thirtyTwo+" = "+Long.toBinaryString(1 << 32));

打印

1 = 1
1073741824 = 1000000000000000000000000000000
-2147483648 = 1111111111111111111111111111111110000000000000000000000000000000
1 = 1

这对我来说没有意义。 是一个64位数字 - 而它似乎在上面的行为就像一个。我知道bitshifteds正在经历int推广,但我不明白在这种情况下发生了什么。longintbyte

任何关于这里发生的事情的指针都会很好:)

感谢

编辑:感谢所有的答案 - 我意识到发生了什么,只要我点击“提交”,但SO进入模式,我无法删除!非常感谢!readonly


答案 1

这是因为 1 是文字,因此应用于整数。结果被转换为 a,但到那时为时已晚。int<<long

如果你写1L<<32,等等,那么一切都会好起来的。 用于表示文字。Llong


答案 2

首先:

Java 语言规范第 15.19 段

换班操作员

如果左侧操作数的升级类型为 int,则仅将右侧操作数的五个最低阶位用作平移距离。就好像右边的操作数受制于按位逻辑 AND 运算符 &(§15.22.1),掩码值0x1f (0b11111)。因此,实际使用的移动距离始终在 0 到 31(包括 0 到 31)的范围内。

如果左侧操作数的升级类型很长,则仅使用右侧操作数的六个最低阶位作为平移距离。就好像右边的操作数受到按位逻辑 AND 运算符 & (§15.22.1) 的影响,掩码值0x3f (0b111111)。因此,实际使用的移动距离始终在 0 到 63(包括 0 到 63)的范围内。

那么,为什么它的工作原理就像你正在移动s而不是s一样?因为你正在改变s!请尝试改用文字:intlongintlong

long thirtyTwo = 1L << 32;

推荐