位运算,即对数据的二进制形式按位进行运算操作,c++中有多种位运算操作:
由于位运算是直接对内存中二进制数据进行操作,不需要进行转化,因此效率很高,速度比+-*/等算数运算更快
C语言中 位运算速度 > +-速度 > */速度 > %速度
合理利用位运算操作可以一定程度上提高程序运行速度,从而避免TLE
壹.左移/右移:
<< 二进制左移(SHL)运算符:
将一个运算对象的各二进制位全部左移若干位,右边补0,超出对应类型范围时左边的位自动丢弃。
例:
printf("%d", 3 << 2);//结果为12
(3)10 = (11)2
*(3)10表示十进制数3,(11)2表示二进制数11
向左移动两位后,右边补0,得到 (1100)2 = (1*23+1*22+0*21+0*20)10 = 12
不难看出: (1100)2 = 十进制下 22*(1*21+1*20) = 22*(3)
以此类推: 3<<k = 2k * 3 ;
同理:n<<k = n* 2^k
因此:我们可以借助 1<<k便捷的计算2的k次方
#由于位运算操做是直接针对内存中的二进制编码进行简单操做,所以位运算的运算效率要高于加减(+ -)操作,更快于乘除(* /)操作,
1<<k,远远快于需要进行k此操作的函数: pow(2,k); 可以减少计算耗时。
在程序时间限制极为紧张时,也可以考虑把一些乘法操作改写成加法操作与左移操作的组合,从而提高计算效率
如: x*10 = x<<3+x<<1;
>> 二进制右移(SHR)运算符:
二进制右移运算符。将一个数的各二进制位全部右移若干位,正数左补0,负数左补1,右边丢弃。
例:
printf("%d", 5 >> 1);//结果为2
printf("%d", -5 >> 1);//结果为-3
(5)10 = (00000101)2
向右移动并舍弃一位,左边补0得 (00000010)2 = (2)10
(-5)10 在计算机中以二进制补码的形式表示为:(11111011)
整体向右移一位,左边补1得:(11111101) = (-3)10
类比左移运算不难看出: n>>k 相当于 n/2^k 向下取整 (注意:对负数向下取整相当于对其绝对值向上取整)
(不建议对负数进行位运算操作)
贰.按位 与 或 取反 异或 运算:
& 按位与(AND)运算符:
按位与操作,按二进制位进行"与"运算,每一位的运算规则于逻辑 “与” && 类似,只有在参与运算两数同时为1时结果为1
即:1与1=1,1与0=0,0与1=0,0与0=0
( 二者位数不一致时向右对齐,较小数左端补0,如: 010 & 111 = 010 )
例:
printf("%d", 7 & 5);//结果为5
(7)10=(111)2 (5)10=(101)2
第一位1&1=1,第二位1&0=0,第三位1&1=1
(111)2 & (101)2 = (101)2 = (5)10
#与运算遵循 交换律,结合律,对或运算的分配律
即 a&b = b&a a&(b&c) = (a&b)&c a&(b|c) = (a&b)|(a&c)
#不难看出,一切正数奇数的二进制最低位一定是1,偶数最低位一定是0
所以可以通过n&1判断n的奇偶性,结果为1则为奇,为0则为偶
因为位运算比取余运算更高效,所以该表达式比n%2耗时更少
#利用&将数的二进制最低非0位变为0
int a = 6;
printf("%d\n", a & a - 1);
6=(110)2 6-1=(101)2 6&5 =(100)2=4
#利用&求数a的二进制中有多少个一 时间复杂度O(k) k为a中1的数量
int a = 6, k = 0;
while (a)
{
a = a & a - 1;
k++;
}
printf("%d\n", k);
每次循环都会将a的最低为1位改为0,并且k++,当所有位全为0时循环结束,可算出a的二进制表示中有多少个1
| 按位或(OR)运算符:
按位或运算符,按二进制位进行"或"运算,每一位的运算规则于逻辑 “或” || 类似,只有在参与运算两数同时为0时结果为0
即:0或0=0,1或0=1,0或1=1,1或1=1
( 二者位数不一致时向右对齐,较小数左端补0,如: 010 | 111 = 111 )
例:
printf("%d", 6 | 9);//结果为15
(6)10=(110)2=(0110)2 (9)10=(1001)2
第一位1|0=1,第二位0|1=1,第三位1|0=1,第四位0|1=1
(110)2 | (1001)2 = (1111)2 = (15)10
#或运算遵循 交换律,结合律,对与运算的分配律
即 a|b = b|a a|(b|c) = (a|b)|c a|(b&c) = (a|b)&(a|c)
~ 按位取反(NOT)运算符:
将二进制位取反,0变为1,1变为0 (对于原码补码的符号位同样生效)
printf("%d", ~3);//结果为-4
(3)10 = (00000011)2 按位取反得到 11111100 是二进制补码形式 转化为十进制得 -4
# ~x+1 = -1*x (利用补码的性质)
^ 按位异或(XOR)运算符:
异或运算符,按二进制位进行"异或"运算。运算规则:当参与运算两数相同时结果为0,不相同时结果为1
即:0^0 = 0 , 1^1 = 1, 0^1 = 1, 1^0 = 1.
( 二者位数不一致时向右对齐,较小数左端补0,如: 010 ^ 111 = 101 )
例:
printf("%d", 2 ^ 3);//结果为1
(2)10=(10)2 , (3)10=(11)2
第一位1^1=0,第二位1^0=1
(10)2 ^ (11)2 = (01)2 = (1)10
#异或遵循:
1、交换律: a^b=b^a
2、结合律: (a^b)^c = a^(b^c)
3、对于任何数x,都有x^x=0,x^0=x
4、自反性 a^b^b = a^0=a
#利用异或交换两数(高效率)
a ^= b;
b ^= a;
a ^= b;
第一步:a ^= b ---> a = (a^b)
第二步:b ^= a ---> b = b^(a^b) ---> b = (b^b)^a = a
第三步:a ^= b ---> a = (a^b)^a = (a^a)^b = b
叁.位运算赋值运算符:
类似于 +=,-=操做,可以将位运算符于等号组合,形成计算并赋值的位运算赋值运算符
<<= | 左移且赋值运算符 | C <<= n 等同于 C = C << n |
>>= | 右移且赋值运算符 | C >>= n 等同于 C = C >> n |
&= | 按位与且赋值运算符 | C &= n 等同于 C = C & n |
^= | 按位异或且赋值运算符 | C ^= n 等同于 C = C ^ n |
|= | 按位或且赋值运算符 | C |= n 等同于 C = C | n |
肆.运算符优先级:
1.括号 | () |
2.按位取反 | ~ |
3.乘除取余 | * / % |
4.加减 | + - |
5.左右移 | << >> |
6.关系运算 | < <= > >= |
7.相等 | == != |
8.按位与 | & |
9.按位异或 | ^ |
10.按位或 | | |
11.逻辑与 | && |
12.逻辑或 | || |
13.三目运算符 | ?: |
14.赋值 | = += -= *= /= %=>>= <<= &= ^= |= |
#可以发现除了~以外,其他位运算符的运算优先级均较低,大致上优先级 算数运算 > 位运算 > 逻辑运算
因此当出现位运算和算数运算(+-*/)相混合的表达式时,应当多加括号以避免运算顺序错误
#不建议背诵此表,只需有大致印象即可,当不确定优先级时,可以通过多加括号解决
# & ^ | 优先级低于<<,因此在c++中可能出现如下错误
cout << 1 ^ 2 << endl;//编译错误
因为 ^ 优先级低于<< ,所以编译器会尝试先执行输出操作,后进行运算,显然计算机是无法执行的,所以编译时编译器会报错,加上括号即可
cout << (1 ^ 2) << endl;