java中“>>”和“>>>”运算符的区别。
在学习它们两者区别的时候最先知道的是: 无符号右移运算符">>>" 右移之后左边位数全部置为0,包括符号位。菜鸟教程上的解释如下图(其中A=0011 1100):
它们对正数操作之后,结果是相同的,但对负数的操作,结果却有很大不同。
对正数操作示例代码和结果如下:
int a=16;
int b=a>>>2;
int c=a>>2;
System.out.println("a>>>2之后二进制表示为:"+Integer.toBinaryString(b));
System.out.println("a>>>2 = "+b);
System.out.println("a>>2之后二进制表示为:"+Integer.toBinaryString(c));
System.out.println("a>>2 = "+c);
对负数操作的示例代码可将a改为 -16 ,其结果如下:
从对负数的操作结果可以明显看出来执行“a>>>2”之后,结果大了好多倍,那为什么会有这种差异呢?在计算机为了统一处理加减法操作,整数都是以补码的形式存储的。因为正数的补码是其本身,所以两种运算符操作之后结果相同。但负数的补码是(符号位保持不变)先取反码,之后在反码的最后一位+1而得到的。
在-16执行 a>>>2,前两位被置0, 符号位发生改变,所以结果变为正数,结果中二进制位数其实是30位。
在-16执行a>>2,为了保持其符号位不变,前两位被置1,然后求得该补码的原码(补码的补码为原码)即为其结果。
总结
两者的区别主要在对负数的操作上,当执行无符号右移运算 >>> 时,移几位左边就填几个0。而执行有符号运算 >> 时,为了保持其符号位不变,移位之后将其置为1。