Java移位运算符

Java 的移位运算符有三种(引用《Java编程思想(Bruce Eckel, 第4版)》):

  • “<<” :左移运算符,按照操作符右侧指定的位数将操作符左边的操作数向左移动(在低位补0);
  • “>>” :右移运算符,按照操作符右侧指定的位数将操作符左边的操作数向右移动;
    “有符号”右移操作符使用“符号扩展”:
    • 若符号为正,则在高位插入0;
    • 若符号为负,则在高位插入1。
  • “>>>”:“无符号”右移操作符,它使用“符号扩展”:无论正负都在高位插入0。
     
  • 没有无符号左移

 这些解释并没有说清楚原理,这里的移位运算符其实是对补码进行操作。


这里需要先回顾一下原码、反码、补码的概念(以32-bit为例,第一位为符号位):

  • 正数: 正数的反码、补码都等于原码,如3,原码、反码、补码都是 0000 0000 0000 0000 0000 0000 0000 0011;
  • 负数:以-3为例
    • 原码:1000 0000 0000 0000 0000 0000 0000 0011;
    • 反码:1111 1111 1111 1111 1111 1111 1111 1100;(符号位不变,其余位取反)
    • 补码:1111 1111 1111 1111 1111 1111 1111 1101;(符号位不变,其余位取反后+1)

下面是几个例子:

  • 正数 
int a = 3;
System.out.println(a << 1);     // 输出: 6
System.out.println(a >> 1);	    // 输出: 1

正数的补码跟原码一样,所以3左移一位即:
    0000 0000 0000 0000 0000 0000 0000 0011 ==> 0000 0000 0000 0000 0000 0000 0000 0110
    即 3 ==> 6 (10进制);
3右移一位:
    0000 0000 0000 0000 0000 0000 0000 0011 ==> 0000 0000 0000 0000 0000 0000 0000 0001
    即 3 ==> 1 (10进制);

  • 负数:
int b = -3;
System.out.println(b << 1);     // 输出: -6
System.out.println(b >> 1);	    // 输出: -2

-3的补码是 1111 1111 1111 1111 1111 1111 1111 1101;
-3左移一位:
    补码变为:1111 1111 1111 1111 1111 1111 1111 1010;
    对补码再取补,得到对应的原码。
    对应原码:1000 0000 0000 0000 0000 0000 0000 0110;
    即 -3 ==> -6 (10进制);
-3右移一位:
    补码变为:1111 1111 1111 1111 1111 1111 1111 1110;
    对补码再取补,得到对应的原码。
    对应原码:1000 0000 0000 0000 0000 0000 0000 0010;
    即 -3 ==> -2 (10进制);

  • 无符号右移
int a = 3;
int b = -3;
System.out.println(a >>> 1);    // 输出: 1
System.out.println(b >>> 1);	// 输出: 2147483646 

无符号右移其实就是将符号位也当成一个普通的位,因为正数的符号位为0,所以对正数来说,无符号右移跟有符号右移是一样的。而对负数来说,无符号右移是在高位插入0,而有符号右移是在高位插入0。

  • -3无符号右移1位过程:

    • 原码:1000 0000 0000 0000 0000 0000 0000 0011;
    • 补码:1111 1111 1111 1111 1111 1111 1111 1101;
    • 无符号右移:0111 1111 1111 1111 1111 1111 1111 1110;
    • 再求补转原码:0111 1111 1111 1111 1111 1111 1111 1110;
      注意: 此时在转换成10进制时,仍认为最高位为符号位。而右移后最高位为0,即正数,而对正数再求补仍是这个数。

    即 -3 ==> 2147483646 (10进制)。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值