位运算

&:按位与

二进制位均为1,则为1。

举例:

1001 & 1000 = 1000

 

|:按位或

二进制位有一个为1,则为1。

举例:

1001 & 1000 = 1001

 

^:按位异或

二进制位不同为1。

举例:

1001 ^ 1000 = 0001

 

~:取反

按位取反,为一元运算符。

举例:

~1001 = 0110

 

<<:左移运算符

一个数的二进制位全部左移N位,右位补0。

举例:

1001 << 1 = 10010

(a<<b,实际上为a * 2^b)

 

>>:右移运算符

一个数的二进制位全部右移N位,左位补0还是1取决于计算机系统。

对于无符号数,左位补0。

对于有符号数,有机器将对左位用符号位填补(即“算术移位”),有机器则对左位用0填补(即“逻辑移位”)。即:如果原来符号位为0(正数),则左位补0。如果符号位原来为1(负数),则左位补0还是1,要取决于所用的计算机系统。

当a是正整数时,a>>b == a / (2^b)

当a是负整数时,a>>b == a / (2^b) 上取整。

 

位运算技巧

技巧一:N&(N-1)可以消去N的最后一位的1

应用:

用O(1)时间检测整数n是否是2的幂次(N的二进制表示中只有一个1,且N>0)

二进制表示中有多少个1(循环消1即可)

将整数A转换为B,需要改变多少个bit位(A和B有多少个BIT位不相同,A异或B之后这个数中1的个数)

 

技巧二:子集枚举

对于大小为N的数组,判断N个二进制位中每一位是0还是1,1表示包括该位置数,0表示不包括,可生成2^N个子集

 

技巧三:a^b^b=a

应用:

只有一个数出现一次,剩下都出现两次,找出出现一次的(所有的数异或可以得到唯一的数)

只有一个数出现一次,剩下都出现三次,找出出现一次的(如果只出现一次的数在该二进制位为1,那么这个二进制位在全部数字中出现次数无法被3整除)

只有两个数出现一次,剩下都出现两次,找出出现一次的(所有的元素异或的结果就是res=x^y。并且res!=0,那么我们可以找出res二进制表示中的某一位是1,那么这一位1对于这两个数x,y只有一个数的该位置是1。对于原来的数组,我们可以根据这个位置是不是1就可以将数组分成两个部分。求出x,y其中一个,就能求出两个了。)

1 #include<stdio.h>
 2 
 3 void findNum(int *a,int n)
 4 {
 5     int ans=0;
 6     int pos=0;
 7     int x=0,y=0;
 8     for(int i=0;i<n;i++)
 9         ans^=a[i];
10     int tmp=ans;
11     while((tmp&1)==0){
12     //终止条件是二进制tmp最低位是1
13             pos++;
14             tmp>>=1;
15     }
16     for(int i=0;i<n;i++){
17         if((a[i]>>pos)&1){//取出第pos位的值
18             x^=a[i];
19         }
20     }
21     y=x^ans;
22     if(x>y) swap(x,y);//从大到小输出x,y
23     printf("%d %d\n",x,y);
24 }
25 int main()
26 {
27     int a[8]={1,2,2,3,4,4,5,3};
28     findNum(a,8);
29 }

 

技巧总结

功能 示例 位运算公式
去掉最后一位 101101  ->  10110 x >> 1
最后加一个0 101101  ->  1011010 x << 1
最后加一个1 101101  ->  1011011 (x << 1) + 1
最后一位变成0 101101  ->  101100 (x | 1) - 1
最后一位变成1 101100  ->  101101 x | 1
最后一位取反 101101  ->  101100 x ^ 1
右边第k位变成0 101101  ->  101001     k=3 x | (1 << (k-1))
右边第k位变成1 101001  ->  101101     k=3 x &~ (1 << (k-1))
右边第k位取反 101101  ->  101001     k=3 x ^ (1 << (k-1))
取最后三位 101101  ->  101 x & 0b111
取最后k位 101101  ->  1101        k=4 x & ((1<<k) - 1)
取右边第k位 101101  ->  1             k=4 x>>(k - 1) & 1
右边k位变成1 101101  ->  101111    k=4 x | ((1<<k) - 1)
右边k位取反 101101  ->  100010 x ^ ((1<<k) - 1)
右边连续的1变成0 100101111  ->  100100000 x & (x + 1)
右边连续的0变成1 101101000  ->  101101111 x | (x - 1)
右边第一个0变成1 101101  ->  101111 x | (x + 1)
取右边连续的1 101111  ->  1111 (x ^ (x + 1)) >> 1
去掉右边第一个1的左部分 101101000  ->  1000 x & (-x)

 

 

负数的二进制表示

计算机中:负数以原码的补码形式表达

 

为何101101000 -> 1000   =====    x & (-x) ?

 

296:00000000 00000000 00000001 00101000

296的原码:00000000 00000000 00000001 00101000(正数,按照绝对值大小转换成的二进制数)

-296的原码:10000000 00000000 00000001 00101000(负数按照绝对值大小转换成的二进制数,然后最高位补1)

 

反码:

正数的反码与原码相同。296的反码:00000000 00000000 00000001 00101000

负数的反码为对原码除符号位外取反。-296的反码:11111111 11111111 11111110 11010111

 

补码:

正数的补码与原码相同。296的补码:00000000 00000000 00000001 00101000

负数的补码为对原码除符号位外各位取反,然后在最后一位加1。-296的补码:11111111 11111111 11111110 11011000

 

那么-296在计算机中表示为:11111111 11111111 11111110 11011000

 

因此:296 & (-296) = 8      100101000 & 011011000   =(0b1000)

 

 

 

posted @ 2021-04-12 18:18  actortoday  阅读(327)  评论(0编辑  收藏  举报