• Java位运算:位异或运算、位与运算、位或运算、位取反运算、左位移运算、右位移运算、无符号右移运算、不用额外变量交换两个整数的值(使用位异或运算)


    1. 预备知识

    位运算符比算术运算符速度要快,但是为什么会快呢?

    因为算术运算符也是通过计算机通过位运算调用栈来实现的。

    对于有符号数,一般在最高位保留符号位,最高位1表示负数,最高位0表示正数。


    1.1 位异或运算(^)

    在介绍下面的题目之前,我们来看看位异或运算符^,其运算规则是:参与运算的数字,低位对齐,高位不足的补零,如果对应的二进制位相同(同时为 0 或同时为 1)时,结果为 0;如果对应的二进制位不相同,结果则为 1。

    下面是的异或运算过程:异或运算你可以理解为不进位的加法运算

      1 1 0 1 0
      1 0 0 0 1
    --------------
    ^ 0 1 0 1 1
    
    • 1
    • 2
    • 3
    • 4

    异或运算满足下面的规律

    • 0 ^ N = 0
    • N ^ N = 0
    • a ^ b = b ^ a
    • (a ^ b) ^ c = a ^ (b ^ c)

    Java代码演示:

    在这里插入图片描述


    1.2 位与(&)运算

    位与运算符为&,其运算规则是:参与运算的数字,低位对齐,高位不足的补零,如果对应的二进制位同时为 1,那么计算结果才为 1,否则为 0。

    规律:0 & N = 0

    示例

      1 1 0 1 0
      1 0 0 0 1
    --------------
    & 1 0 0 0 0
    
    • 1
    • 2
    • 3
    • 4

    Java代码演示:

    在这里插入图片描述


    1.3 位或(I)运算

    位或运算符为I,其运算规则是:参与运算的数字,低位对齐,高位不足的补零。如果对应的二进制位只要有一个为 1,那么结果就为 1;如果对应的二进制位都为 0,结果才为 0。

    规律:1 | N = 1

    示例

      1 1 0 1 0
      1 0 0 0 1
    --------------
    | 1 1 0 1 1
    
    • 1
    • 2
    • 3
    • 4

    Java代码演示:

    在这里插入图片描述


    1.4 位取反(~)运算

    位取反运算符为~,其运算规则是:只对一个操作数进行运算,将操作数二进制中的 1 改为 0,0 改为 1。

    示例

      1 1 0 1 0
    --------------
    ~ 0 0 1 0 1
    
    • 1
    • 2
    • 3

    示例:

    在这里插入图片描述


    1.5 左位移(<<)运算

    左移位运算符为 <<,其运算规则是:按二进制形式把所有的数字向左移动对应的位数,高位移出(舍弃),低位的空位补零。

    规律:左移n位相当于乘以2n(不溢出的情况下)

    示例:

        0 0 1 0  左移2位-->  1 0 0 0
       10进制:2             10进制:8 
       2左移2位相当于:2 * 2^2 = 2 * 4 = 8 (不溢出的情况下)
    
    • 1
    • 2
    • 3

    Java代码演示:

    在这里插入图片描述


    1.6 右位移(>>)运算

    右位移运算符为 <<,其运算规则是:按二进制形式把所有的数字向右移动对应的位数,低位移出(舍弃),高位的空位补零。

    规律:左移n位相当于除以2n(不溢出的情况下)

    示例:

        1 0 0 0  左移2位-->  0  0 1 0 
       10进制:8             10进制:2
       8左移2位相当于:8 / 2^2 = 8 / 4 =  2 (不溢出的情况下)
    
    • 1
    • 2
    • 3

    Java代码演示:

    在这里插入图片描述


    1.6 无符号右移(>>>)运算

    Java提供了两种右移运算符:>> 和 >>>。其中,>> 被称为有符号右移运算符,>>> 被称为无符号右移运算符,它们的功能是将参与运算的对象对应的二进制数右移指定的位数。二者的不同点在于 >> 在执行右移操作时,若参与运算的数字为正数,则在高位补0;若为负数,则在高位补1。而 >>> 则不同,无论参与运算的数字为正数或为负数,在执运算时,都会在高位补0。

    Java代码演示:

    在这里插入图片描述


    1.7 提示:Java中没有无符号左移这个说法

    左移运算没有有符号和无符号左移动,在左移时,移除高位的同时在低位补0。(左移不保留符号位,和无符号左移是一样的,所以没必要加一个无符号左移,一直左移会变成0


    2. 不用额外变量交换两个整数的值

    我们平时交换2个数字都是这样写的:这样做用了用了一个额外的中间变量temp。

    int temp = a;
    a = b;
    b = temp;
    
    • 1
    • 2
    • 3

    在有了上面的基础知识后,你就可以开始思考如何做到不用额外变量交换两个整数的值?话不多说,我们直接公布答案:

    a = a ^ b;
    b = a ^ b;
    a = a ^ b;
    
    • 1
    • 2
    • 3

    你可能还有点不相信,来让我们让事实说话:

        @Test
        public void test() {
            int a = 520;
            int b = 1314;
            System.out.println("a before: " + a);
            System.out.println("b before: " + b);
    
            // 使用 ^ 运算交换2个整数的值
            a = a ^ b;
            b = a ^ b;
            a = a ^ b;
    
            System.out.println("a after: " + a);
            System.out.println("b after: " + b);
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    运行结果:
    在这里插入图片描述

    你可能现在满脸都写着不可思议,下面让我们来分析使用 ^ 运算交换2个整数的值,不需要额外变量的原因:

    假设 a = x; b = y;

    a = a ^ b; // 这一步后 a = x ^ y; b = y;
    b = a ^ b; // 这一步后 a = x ^ y; b = x ^ y ^ y = x ^ ( y ^ y) = x ^ 0 = x;
    a = a ^ b; // 这一步后 a = x ^ y ^ x = (x ^ x) ^ y = 0 ^ y = y; b = x;
    
    • 1
    • 2
    • 3

    我想现在你应该明白了。



  • 相关阅读:
    C语言画直方图
    Netty 学习(六):创建 NioEventLoopGroup 的核心源码说明
    插片式远程 I/O模块:热电阻温度采集模块与PLC配置案例
    足不出户就能购遍全球?亚马逊中国Prime会员日即将开启
    【漏洞复现】SpringBlade dict-biz SQL注入漏洞
    【FFmpeg】视频与图片互相转换 ( 视频与 JPG 静态图片互相转换 | 视频与 GIF 动态图片互相转换 )
    【答读者问】把Go基础学完后,是学web方向还是区块链方向?
    查询sqlserver内存分配情况的SQL
    QoS策略
    JAVA的分页列表+搜索
  • 原文地址:https://blog.csdn.net/I_r_o_n_M_a_n/article/details/125307879