Java语法学习4

亦或^运算除了交换之外leetcode上还有一道有趣的算法题,其中有一位大牛发布了让别人惊讶的源码,短短几行,巧妙运用位运算亦或的特性。

// 存在奇数次的数是哪个
class Bit {
    public static void main(String[] args) {
        //^异或
        //leetcode.com
        int[] n = {9,4,9,5,4,5,7};

        int r = 0;
        for(int e : n) {
            r ^= e;
        }
        System.out.println(r);
    }
}

输出7,当然还有很多之类的变形,例如找出出现3次的,4次的。

位运算是效率很高的运算,都是数值转换为二进制运算

~按位取反

    byte b = 127;

   System.out.print(~b);

打印出-128;

按位取反,再加上1得原数负值。

0011 

------

1100

还要加1

1101

因为第一位是符号,1表示是负数

则-5


位移符号

int a = 3;

a<<2 a*2^2;

64 >> 3 64/2^3;

如果位不够 3 >>7

0000 0011 , 从又开始删去7位剩下0,从右补齐0,补齐7个零。

如果是负数移多了位,则为-1,(向左移)


下面特此为取反运算符计算方法归纳:(计算机组成原理和计算机导论)

“~”运算符在c、c++、java、c#中都有,之前一直没有遇到这个运算符,今天碰到一道笔试题,经过学习,终于搞懂了。

要弄懂这个运算符的计算方法,首先必须明白二进制数在内存中的存放形式,二进制数在内存中是以补码的形式存放的

  另外正数和负数的补码不一样,正数的补码,反码都是其本身,既:

 正数9(1001)在内存中存储为01001,必须补上符号位

补码为01001

反码为01001,其中前面加的0是符号位,负数的是1

负数 -1(10001,开头的1为符号位)在内存中存放为,11111

其中11111为其补码

反码为11110

负数的补码是:符号位不变,其余各位求反,末位加1 ,既11111

反码是:符号位为1,其余各位求反,但末位不加1 ,既11110

也就是说,反码末位加上1就是补码 

------------------------------------------------------------------------------------------------------------------------------------

弄懂了上述情况后,如何计算就好办了

假设有一个数~9,计算步骤如下,9=1001

其补码为01001,

对其取反10110,这个数是在内存中存放的,现在需要装换成原码,既先减1,然后取反得

                11010,符号位为1是负数,既-10

另外在Java存储中,溢出都是走圆圈下一位,-126,-127,-128,127,126,125,....3,2,1,0,-1,-2,-3,-125,-126



原文地址:https://www.cnblogs.com/zhangmingzhao/p/7256707.html