public class Shifty {
public static void main(String[] args) {
Int i = 0;
while (-1 << i != 0)
i++;
System.out.println(i);
}
}
常量-1 是所有32 位都被置位的int 数值(0xffffffff)。左移操作符将0 移入到由移位所空出的右边的最低位,因此表达式(-1 << i)将i 最右边的位设置为0,并保持其余的32 - i 位为1。很明显,这个循环将完成32 次迭代,因为-1 << i 对任何小于32 的i 来说都不等于0。你可能期望终止条件测试在i 等于32 时返回false,从而使程序打印32,但是它打印的并不是32。实际上,它不会打印任何东西,而是进入了一个无限循环。
这条规则作用于全部的三个移位操作符:<<、>>和>>>。移位长度是对32取余的,所以总是介于0 到31 之间;如果左操作数是long 类型的,则对64 取余,介于0 到63 之间。如果试图对一个int 数值移位32 位,或者是对一个long 数值移位64 位,都只能返回这个数值自身的值。没有任何移位长度可以让一个int 数值丢弃其所有的32 位,或者是让一个long数值丢弃其所有的64 位。
(-1 << 32) == -1;
如果可能的话,移位长度应该是常量。如果移位长度紧盯着你不放,那么你让其值超过31,或者如果左操作数是long类型的,让其值超过63 的可能性就会大大降低。当然,你并不可能总是可以使用常量的移位长度。当你必须使用一个非常量的移位长度时,请确保你的程序可以应付这种容易产生问题的情况,或者压根就不会碰到这种情况。
很多程序员都希望具有负的移位长度的右移操作符可以起到左移操作符的作用,反之亦然。但是情况并非如此。右移操作符总是起到右移的作用,而左移操作符也总是起到左移的作用。总之,移位长度是对32 取余的,或者如果左操作数是long 类型的,则对64 取余。因此,使用任何移位操作符和移位长度,都不可能将一个数值的所有位全部移走。同时,我们也不可能用右移操作符来执行左移操作,反之亦然。如果可能的话,请使用常量的移位长度,如果移位长度不能设为常量,那么就要千万当心。转载自: 无码团队blog-wuma.koubei.com