1、 左移运算符<<
左移运算符<<使指定值的所有位都左移指定的次数。
1)通用格式
value << numnum 指定要移位值value 移动的位数。
左移的规则只记住一点:丢弃最高位,0补最低位
如果移动的位数超过了该类型的最大位数,那么编译器会对移动的位数取模。如对int型移动33位,实际上只移动了33%32=1位。
2)运算规则
按二进制形式把所有的数字向左移动对应的位数,高位移出(舍弃),低位的空位补零。当左移的运算数是int 类型时,每移动1位它的第31位就要被移出并且丢弃;
当左移的运算数是long 类型时,每移动1位它的第63位就要被移出并且丢弃。
当左移的运算数是byte 和short类型时,将自动把这些类型扩大为 int 型。
3)数学意义
左移运算符<<使指定值的所有位都左移规定的次数。2、 右移运算符>>
右移运算符<<使指定值的所有位都右移指定的次数。1)通用格式
value >> numnum 指定要移位值value 移动的位数。
右移的规则只记住一点:符号位不变,左边补上符号位
2)运算规则
按二进制形式把所有的数字向右移动对应的位数,低位移出(舍弃),高位的空位补符号位,即正数补零,负数补1当右移的运算数是byte 和short类型时,将自动把这些类型扩大为 int 型。
例如,如果要移走的值为负数,每一次右移都在左边补1,如果要移走的值为正数,每一次右移都在左边补0,这叫做符号位扩展(保留符号位)(sign extension ),在进行右移
操作时用来保持负数的符号。
3)数学意义
右移一位相当于除2,右移n位相当于除以2的n次方。3、无符号右移>>>
1) 通用格式
value >>> numnum 指定要移位值value 移动的位数。
2) 运算规则
无符号右移的规则只记住一点:忽略了符号位扩展,0补最高位无符号右移运算符>>> 只是对32位和64位的值有意义
==================================我是分割线===============================================================
下面是写的一组测试:
<span style="font-size:14px;"> @Test
public void test(){
//左移<<规则:移除的高位丢弃,低位补0 , 左移n位就相当于在原来值的基础上乘以2的n次方
short v1 = -3;
//原码:1000 0000 0000 0000 0000 0000 0000 0011
//取反~
//反码:1111 1111 1111 1111 1111 1111 1111 1100
//加1
//补码:1111 1111 1111 1111 1111 1111 1111 1101
//左移3位后
//补码:1111 1111 1111 1111 1111 1111 1110 1000
//减1
//反码:1111 1111 1111 1111 1111 1111 1110 0111
//取反
//原码:1000 0000 0000 0000 0000 0000 0001 1000 ->值:-24
Assert.assertTrue(v1<<3 == -24);
short v2 = 3;
//正数原码,反码,补码相同
//原码: 0000 0000 0000 0000 0000 0000 0000 0011
//反码: 0000 0000 0000 0000 0000 0000 0000 0011
//补码: 0000 0000 0000 0000 0000 0000 0000 0011
//补码: 0000 0000 0000 0000 0000 0000 0001 1000
//反码: 0000 0000 0000 0000 0000 0000 0001 1000
//原码: 0000 0000 0000 0000 0000 0000 0001 1000 ->值:24
Assert.assertTrue(v2<<3 == 24);
short v3 = -100;
//原码: 1000 0000 0000 0000 0000 0000 0110 0100
//取反~
//反码: 1111 1111 1111 1111 1111 1111 1001 1011
//加1
//补码: 1111 1111 1111 1111 1111 1111 1001 1100
//左移3位后
//补码: 1111 1111 1111 1111 1111 1100 1110 0000
//减1:
//反码: 1111 1111 1111 1111 1111 1100 1101 1111
//取反~
//原码: 1000 0000 0000 0000 0000 0011 0010 0000 ->值:-800
Assert.assertTrue(v3<<3 == -800);
//右移>>规则:移除的低位丢弃,高位根据原符号位补相应的值,如原为负值,符号位为1,则补1,否则补0
//-3
//原码:1000 0000 0000 0000 0000 0000 0000 0011
//取反~
//反码:1111 1111 1111 1111 1111 1111 1111 1100
//加1
//补码:1111 1111 1111 1111 1111 1111 1111 1101
//右移3位,高位补1
//补码:1111 1111 1111 1111 1111 1111 1111 1111
//减1
//反码:1111 1111 1111 1111 1111 1111 1111 1110
//取反~
//原码:1000 0000 0000 0000 0000 0000 0000 0001 ->值:-1
Assert.assertTrue(v1>>3 == -1);
//3
//正数原码,反码,补码相同
//原码: 0000 0000 0000 0000 0000 0000 0000 0011
//反码: 0000 0000 0000 0000 0000 0000 0000 0011
//补码: 0000 0000 0000 0000 0000 0000 0000 0011
//右移3位,高位补0
//补码: 0000 0000 0000 0000 0000 0000 0000 0000
//反码: 0000 0000 0000 0000 0000 0000 0000 0000
//原码: 0000 0000 0000 0000 0000 0000 0000 0000 ->值:0
Assert.assertTrue(v2>>3 == 0);
//-100
//原码: 1000 0000 0000 0000 0000 0000 0110 0100
//反码: 1111 1111 1111 1111 1111 1111 1001 1011
//补码: 1111 1111 1111 1111 1111 1111 1001 1100
//左移3位,高位补1
//补码: 1111 1111 1111 1111 1111 1111 1111 0011
//反码: 1111 1111 1111 1111 1111 1111 1111 0010
//原码: 1000 0000 0000 0000 0000 0000 0000 1101 ->值:-13
Assert.assertTrue(v3>>3 == -13);
//右移>>>规则:移除的低位丢弃,高位补0
//-3
//原码:1000 0000 0000 0000 0000 0000 0000 0011
//取反~
//反码:1111 1111 1111 1111 1111 1111 1111 1100
//加1
//补码:1111 1111 1111 1111 1111 1111 1111 1101
//右移3位,高位补0
//补码:0001 1111 1111 1111 1111 1111 1111 1111
//符号位为0,则表正数,补码和反码相同
//反码:0001 1111 1111 1111 1111 1111 1111 1111
//符号位为0,则表正数,原码和反码相同
//原码:0001 1111 1111 1111 1111 1111 1111 1111 ->值:536870911
Assert.assertTrue(v1>>>3 == 536870911);
//3
//原码: 0000 0000 0000 0000 0000 0000 0000 0011
//反码: 0000 0000 0000 0000 0000 0000 0000 0011
//补码: 0000 0000 0000 0000 0000 0000 0000 0011
//右移3位,高位补0
//补码: 0000 0000 0000 0000 0000 0000 0000 0000
//反码: 0000 0000 0000 0000 0000 0000 0000 0000
//原码: 0000 0000 0000 0000 0000 0000 0000 0000 ->值:0
Assert.assertTrue(v2>>>3 == 0);
//-100
//原码: 1000 0000 0000 0000 0000 0000 0110 0100
//反码: 1111 1111 1111 1111 1111 1111 1001 1011
//补码: 1111 1111 1111 1111 1111 1111 1001 1100
//左移3位,高位补0
//补码: 0001 1111 1111 1111 1111 1111 1111 0011
//符号位为0,则表正数,补码和反码相同
//反码: 0001 1111 1111 1111 1111 1111 1111 0011
//符号位为0,则表正数,原码和反码相同
//原码: 0001 1111 1111 1111 1111 1111 1111 0011 ->值:536870899
Assert.assertTrue(v3>>>3 == 536870899);
}</span>