位运算——优选算法

一、基础位运算

按位与:&

两者为真(1)才为真(1),也可以理解为只要有假(0)就一定为假(0),如下:

按位或:|

只要有一个是真(1)则为真(1),否则为假(0)

按位异或:^

相同为假(0),相异为真(1),如下:

异或(^)运算规律:

对任意常数a,b,c有:

a^0=a,a^a=0,a^b^c=a^(b^c)

按位取反:~

真(1)变为假(0),假(0)变为真(1),如下:

位运算的优先级

关于位运算的优先级是一个让人很头疼的事情,不过也很好解决,直接简单粗暴多加括号就行,就像写宏定义一样。

二、进阶位运算

1.确定n的第x位是0还是1

把n右移x位,然后去&1,如下:

(n>>x)&1

如果结果为0则第x位为0,如果结果为1则第x位为1。

2.把n的x位改成1

把1左移x位,然后去| n,如下:

n=(1>>x)|n 或 n|=(1>>x)

3.把n的x位改成0

把1左移x位,然后按位取反后去&n,如下:

n=(~(1>>x))&n 或 n&=(~(1>>x))

4.提取二进制最右侧的1

(~n+1)&n

5.去掉二进制最右侧的1

n&(n-1)

相关题目1:231. 2 的幂 - 力扣(LeetCode)

解题代码:

cpp 复制代码
class Solution {
public:
    bool isPowerOfTwo(int n) {
        return n>0&&(n&(n-1))==0;
    }
};

相关题目2:191. 位1的个数 - 力扣(LeetCode)

解题代码:

cpp 复制代码
int hammingWeight(uint32_t n)
{
    int t=0;
    while(n)
    {
        n&=n-1;
        t++;
    }
    return t;
}

相关题目3:461. 汉明距离 - 力扣(LeetCode)

解题代码:

cpp 复制代码
class Solution {
public:
    int hammingDistance(int x, int y)
    {
        int z=x^y;
        int sum=0;
        for(int i=0;i<32;i++)
            if((z>>i)&1==1) sum++;
        return sum;
    }
};

三、位运算的应用

137. 只出现一次的数字 II - 力扣(LeetCode)

需要注意这个题用暴力方法是很好想的,但是要求使用线性的时间复杂度和常级别的空间来解决该问题。

算法思路:

  1. 开辟一个32大小的int类型空间arr
  2. 把数组中的所有数的二进制位累加到一个arr的对应位置上,比如把所有元素二进制的第1位累加到arr[0]位置,所有元素二进制的第2位,累加到arr[1]位置。
  3. 那么arr[i]%3得到的就是只出现一次的那个元素的二进制的第i+1位。
  4. 定义一个变量m=0用来储存结果,把m的二进制位的对应位置改成arr[i]%3。

解题代码:

cpp 复制代码
class Solution {
public:
    int singleNumber(vector<int>& nums) {
        int arr[32]={0};
        for(int i=0;i<nums.size();i++)
        {
            for(int j=0;j<32;j++)
                arr[j]+=1&(nums[i]>>j);
        }
        int m=0;
        for(int i=0;i<32;i++)
        {
            m|=(arr[i]%3)<<i;
        }
        return m;
    }
};

面试题 17.19. 消失的两个数字 - 力扣(LeetCode)

算法思路:

记:缺失的两个数字为ret1和ret2,初始化为0。

  1. 定义变量k=0,将k与1到N的数字异或(^)然后与数组中的所有元素异或(^)
  2. 通过异或(^)的规律我们可以知道异或后的结果k就是缺失的两个数字的异或结果。
  3. 找到k的二进制中的任意一个为1的第u位。
  4. 那么我们可以知道ret1和ret2的第u位是不一样的,所以我们把数组中的元素和1到N数字,分为两组。第u位为0的为一组,第u位为1的为二组,这样就把数组中缺失的数字分开了,然后让ret1异或上一组的所有元素,ret2异或上二组的所有元素,则就能得到缺失的两个数字ret1和ret2。
cpp 复制代码
class Solution {
public:
    vector<int> missingTwo(vector<int>& nums)
    {
        int k=0,n=nums.size();
        for(int i=0;i<n;i++) k^=nums[i];
        for(int j=1;j<=n+2;j++) k^=j;
        int u=0;
        while(((k>>u)&1)!=1) u++;
        int ret1=0,ret2=0;
        for(int j=0;j<n;j++)
        {
            if(((nums[j]>>u)&1)==1) ret1^=nums[j];
            else ret2^=nums[j];
        }
        for(int j=1;j<=n+2;j++)
        {
            
            if((j>>u)&1==1) ret1^=j;
            else ret2^=j;
        }
        return {ret1,ret2};
    }
};
相关推荐
东风吹柳12 分钟前
观察者模式(sigslot in C++)
c++·观察者模式·信号槽·sigslot
A懿轩A20 分钟前
C/C++ 数据结构与算法【栈和队列】 栈+队列详细解析【日常学习,考研必备】带图+详细代码
c语言·数据结构·c++·学习·考研·算法·栈和队列
Python机器学习AI24 分钟前
分类模型的预测概率解读:3D概率分布可视化的直观呈现
算法·机器学习·分类
Yvemil728 分钟前
《开启微服务之旅:Spring Boot 从入门到实践》(三)
java
Anna。。29 分钟前
Java入门2-idea 第五章:IO流(java.io包中)
java·开发语言·intellij-idea
.生产的驴1 小时前
SpringBoot 对接第三方登录 手机号登录 手机号验证 微信小程序登录 结合Redis SaToken
java·spring boot·redis·后端·缓存·微信小程序·maven
爱上语文1 小时前
宠物管理系统:Dao层
java·开发语言·宠物
吕小明么1 小时前
OpenAI o3 “震撼” 发布后回归技术本身的审视与进一步思考
人工智能·深度学习·算法·aigc·agi
王ASC1 小时前
SpringMVC的URL组成,以及URI中对/斜杠的处理,解决IllegalStateException: Ambiguous mapping
java·mvc·springboot·web
是小崔啊1 小时前
开源轮子 - Apache Common
java·开源·apache