基础算法----位运算

位运算定义

计算机中以二进制的方式来存储东西。位运算指的就是计算机直接对整数在内存中的二进制位进行操作。



常见的位运算


与的定义:二进制下相同位的两个数字都为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游戏

 此处不讲(几句话说不完

欢迎补充



参考资料

1.位运算 - OI Wiki (oi-wiki.org)

2.位运算_百度百科 (baidu.com)

3.《算法竞赛进阶指南》李煜东

如有侵权,请尽快联系好删去内容。



                                                               END!!!

  • 46
    点赞
  • 44
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值