Java运算符 >> 和 >>>
今天再看HashMap源码时,看到>>>
这个运算符,第一眼以为时右移但仔细看比右移>>
运算符要多一个>符号
搜索后了解到>>>
运算符叫 无符号右移运算符,那么他和右移运算符有什么区别?
1.>>右移运算符
这个运算符大家都很熟悉,接触编程基本上不久就会接触,它就是把数值按二进制的形式右移多少位
例如:5 >> 2;
//假设5是byte类型
0000 0101
//空出的两位补0
0000 0001 [01]
= 1
负数右移: -5 >> 2
//负数在计算是以补码的形式所以我们要先求补码,
//-5原码是:
10000101
//保留符号位取反:
11111010
//然后+1,得到补码
11111011
//进行右移,高位补1
11111110 [11]
//然后-1
11111101
//保留符号位,取反
10000010 = -2
2.>>>无符号右移运算符
这个符号是将数值进行右移高位无论正负都进行补0,它会忽略符号位
例如:5 >>> 2 和右移结果一样等于 1
然后:-5>>>2 还是等于 -2 吗?如果是 -5 >>>=2 呢?
public class Test{
public static void main(String[] args){
byte val1 = -5;
System.out.println(val1 >>> 2);
byte val2 = -5;
val2 >>>= 2;
System.out.println(val2);
}
}
运行结果:
1073741822
-2
为什么是两个结果?
//-5 byte类型的补码之前已经求过
11111011
//进行无符号右移2位
00111110
//最高位为0,计算机会认为它是正数结果将是:
62
//但是运算结果为 1073741822 ,离了大普
//这里我猜想是在进行 val1 >>> 2 运算时,将val1转成了int型所以这时val1的真实的二进制值为
11111111 11111111 11111111 11111011
//这时进行右移2位
00111111 11111111 11111111 11111110 [11]
//换成十进制
1073741822
//那么为什么val2 >>>= 2 为 -2也不是我们想要的62呢?
//这里使用 >>>= 运算会将结果转成byte类型,即结果变成
11111110
//此时计算机会认该值为负数的补码,进行求原码操作
//将值-1,保留符号位取反得
10000010
//结果为
-2
但如果使用int类型话,将不会发生这种情况,就很奇怪。