移位操作符对short、byte等类型的操作
阅读《Java编程思想》,碰到关于移位操作符>>>与组合操作符>>>=在操作short、byte等类型时有些差异,思考了很久并上网查了资料才想通,故记录下来。若其中有错误的地方欢迎指正!
书上两段原话
如果对char、byte或者short类型的数值进行移位处理,那么在移位进行之前,它们会被转换为int类型,并且得到的结果也是一个int类型的值。
只有数值右端的低5位才有用。这样可防止我们移位超过int型值所具有的位数。
“移位”可与“等号”(<<=或>>=或>>>=)组合使用。此时,操作符左边的值会移动由右边的值指定的位数,再将得到的结果赋给左边的变量。
但在进行“无符号”右移位结合赋值操作时,可能会遇到一个问题:如果对byte或short值进行这样的移位运算,得到的可能不是正确的结果。
它们会先被转换为int类型,再进行右移操作,然后被截断,赋值给原来的类型,在这种情况下可能得到-1的结果。
书上的代码例子
public static void main(String[] args){
int i = -1;
myPrint(Integer.toBinaryString(i));
i >>>= 10;
myPrint(Integer.toBinaryString(i));
short s = -1;
myPrint(Integer.toBinaryString(s));
s >>>= 10;
myPrint(Integer.toBinaryString(s));
byte b = -1;
myPrint(Integer.toBinaryString(b));
b >>>= 10;
myPrint(Integer.toBinaryString(b));
b = -1;
myPrint(Integer.toBinaryString(b));
myPrint(Integer.toBinaryString(b >>> 10));
}
执行结果
理解
对于short值的>>>=操作:
首先,short的-1的二进制表示为:1111 1111 1111 1111
然后Integer.toBinaryString(s)会把s转为int型
故打印结果为:1111 1111 1111 1111 1111 1111 1111 1111
接下来执行s >>>= 10,根据原话知道,会先进行int转换,然后右移10位
变成:0000 0000 0011 1111 1111 1111 1111 1111
然后再截断赋给s,又变成:1111 1111 1111 1111
打印的时候又被转换为int,故打印结果为:1111 1111 1111 1111 1111 1111 1111 1111
对于short值的>>>操作:
书中例子是byte,此处也以short为例
首先,short的-1的二进制表示为:1111 1111 1111 1111
然后Integer.toBinaryString(s)会把s转为int型
故打印结果为:1111 1111 1111 1111 1111 1111 1111 1111
接下来执行System.out.println(Integer.toBinaryString(s >>> 10));
根据原话知道,会先进行int转换,然后右移10位
变成:0000 0000 0011 1111 1111 1111 1111 1111
然后直接返回int型,故打印结果为:11 1111 1111 1111 1111 1111
总结
其实简单说来,就是>>>=在操作short、byte等类型是最后要进行截断赋值。
By the way,在上面第一段中关于这句话“只有数值右端的低5位才有用。这样可防止我们移位超过int型值所具有的位数。”我没怎么想通,希望理解的能够给我讲解一下,谢谢!