位运算定义
计算机中以二进制的方式来存储东西。位运算指的就是计算机直接对整数在内存中的二进制位进行操作。
常见的位运算
与
与的定义:在二进制下相同位的两个数字都为1,则为1;反之则为0。
与的表达:在c++中通常通过“&“或者"and"来表示;
在此我们以2和3为例来进行与运算:2的二进制表示为 10 ;3的二级制表示为 11;根据与的定义可知: 2(10)&3(11)=2(10);
与的应用:我们可以使用与进行奇数与偶数判定:任何一个数&1为1则为奇数反正为偶数。
//判断奇偶
#include<bits/stdc++.h>
using namespace std;
int main()
{
int n;
cin>>n;
if(n&1==1)
{
cout<<"奇数"<<endl;
}
else
{
cout<<"偶数"<<endl;
}
return 0;
}
或
或的定义:在二进制下相同位的两个数字其中一个为1,则为1;反之则为0。
或的表达:在c++中通常通过“|“或者"or"来表示;
在此我们以2和3为例来进行与运算:2的二进制表示为 10 ;3的二级制表示为 11;根据与的定义可知: 2(10)|3(11)=3(11);
或的应用:我们可以将数或1转化为离其最近的偶数。
异或
异或的定义:在二进制下相同位的两个数字不同时,则为1;反之则为0。
异或的表达:在c++中通常通过”^“或者"xor"来表示在题目中也常用“⊕”来表示;
在此我们以2和3为例来进行与运算:2的二进制表示为 10 ;3的二级制表示为 11;根据与的定义可知: 2(10)|3(11)=1(1);
异或的特征:两次异或操作后为其本身;(原理很好理解)
取反
定义:字面意思:原本该位为“1”则变为“0”,为“0”则变为“1”;
符号“~”
补码:在二进制表示下,正数和 0 的补码为其本身,负数的补码是将其对应正数按位取反后加一。(csp初赛可能会考)
左移与右移
“<<"左移 各二进位全部左移若干位,高位丢弃,低位补0。
">>"右移 各二进位全部右移若干位,对无符号数,高位补0,有符号数,右移补1 11。
注意:虽然左移与右移的定义十分简单,但其在信息学竞赛中是否重要,有以下几个例子
1.1<<k本质为:2的K次方,k<<1的本质为2k
2.1>>k本质为除以2的K次方
3.快速幂运算 例题:P1226 【模板】快速幂 - 洛谷 (luogu.com.cn)
//快速幂
#include<bits/stdc++.h>
using namespace std;
#define ll long long
ll mul(ll a,ll b,ll m)
{
ll res=1;
while(b>0)
{
if(b&1) res=(res*a)%m;
a=(a*a)%m;
b>>=1;
}
return res;
}
位运算的目的
1.加快程序运行时间
转自:C++ 运算符优先级顺序表 (最新 / 完整) - greyqz - 博客园 (cnblogs.com)
通过上述的表格我们可以知道位运算比普通的加减乘除的时间复杂度是要小的(我认为是使用位运算进行操作的根本原因)使用位运算使得我们的常数可以变小从而使得运行时间变小(可能可以使题目从TLE到AC)。
在此列举几个位运算操作:
1.二分法:用(l+r+1)>>1代替mid=(l+r)/2;
2.上文所说的快速幂。
2.二进制状态压缩
1.取出整数n在二进制下的第K位
(n>>k)&1
2.取出整数n在二进制表示下的后K位
n&((1<<k)-1)
3.对整数n在二进制下的第k位赋值1
n|(1<<k)
4.对整数n在二进制下的第k位赋值0
n&(~(1<<k))
5.对整数n在二进制下的第k位取反
n xor (1<<k)
3.博弈论中的Nim游戏
此处不讲(几句话说不完)
欢迎补充
参考资料
3.《算法竞赛进阶指南》李煜东
如有侵权,请尽快联系好删去内容。
END!!!