【优先算法】专题——位运算

news2025/2/4 22:42:50

在讲解位运算之前我们来总结一下常见的位运算

一、常见的位运算

1.基础为运算

<<      &:有0就是0

>>      |:有1就是1

~        ^:相同为0,相异位1 /无进位相加


2.给一个数 n,确定它的二进制表示中的第x 位是0还是1

n:0 1 1 0 1 0 1 0 0 1

(n >> x) & 1


3.将一个数n的二进制表示的第x位修改成1

               x

0 1 1 0 1 0 1 0 1 1

0 0 0 0 0 1 0 0 0 0

0 1 1 0 1 1 1 0 1 1(使用|)

n |=(1 << x)

n = n | (1 << x)


4.将一个数n的二进制表示的第x位修改成0

            x

0 1 1 0 1 0 1 1 0 0

1 1 1 1 0 1 1 1 1 1(取反得到:0 0 0 0 0 1 0 0 0 0)

0 1 1 0 0 0 1 1 0 0(使用&)

n &= (~(1<<x))

n = n &(~(1<<x))


5.位图思想

我们可以使用一个哈希表来存储我们的信息方便我们增删查改主要还是为了 我们查找因为可以使用O(1)的时间复杂度来查找,但是现在我们可以使用一个int变量来进行,一个int类型4个字节32个bit位,我们可以用某一位bit位上的0或者1来表示我们的信息,0表示一个信息1表示一个信息,本质还是一个哈希表。

位图会大量用到我们2、3、4这几个操作,专门为位图服务


6.提取一个数(n)二进制中最右侧的1

n & -n 将最右侧的1,左边的区域全部变成相反

0 1 1 0 1 0 1 0 0 0(原码)

1 0 0 1 0 1 0 1 1 1(反码)

1 0 0 1 0 1 1 0 0 0(+1,补码)

0 1 1 0 1 0 1 0 0 0(原码)

0 0 0 0 0 0 1 0 0 0(原码和补码进行&)


7.干掉一个数(n)二进制表示中最右侧的1

n & (n-1)将最右侧的1,右边的区域(包含1)全部变成相反

n :   0 1 1 0 1 0 1 0 0

n -1:0 1 1 0 1 0 0 1 1

&        0 1 1 0 1 0 1 0 0

___________________

           0 1 1 0 1 0 0 0 0


8.位运算的优先级

能加括号就加括号


9.异或(^)运算的运算律

1.a ^ 0 = a

2.a ^ a = 0(消消乐)

3.a ^ b ^ c = a ^(b ^ c)

一个奇数,一堆偶数最终的结果为奇数,因为偶数抵消了为0


通过上面的总结我们可以尝试写一下如下五个题

191.位一个的个数

题目链接:位一个的个数

题目描述:

参考代码:
class Solution {
public:
    int hammingWeight(int n) {
       int count = 0;
       while(n != 0) 
       {
            count++;
            n = n & (n-1);//把最低位1的右边互为相反数(包含1)
       }
       return count;
    }
};

338.比特位计数

题目链接:比特位计数

题目描述:

参考代码:
class Solution {
public:
    vector<int> countBits(int n) {
        vector<int>ans(n+1,0);
        for(int i = 1;i <= n;i++)
        {
           ans[i] = ans[i>>1] + (i & 1);
        }
        return ans;
    }
};

461.汉明距离

题目链接:汉明距离

题目描述:

参考代码:
//对应的位置值不相同的个数。例如,假设有两个十进制数a=93和b=73,
// 如果将这两个数用二进制表示的话,有
// a=1011101
// b=1001001,
// 可以看出,二者的从右往左数的第3位、第5位不同(从1开始数)
// 因此,a和b的汉明距离是2。
class Solution {
public:
    int hammingDistance(int x, int y) {
        int s = x ^ y, ret = 0;
        while (s)
        {
            ret += s & 1;
            s >>= 1;
        }
        return ret;
    }
};

136.只出现一次的数字

题目链接:只出现一次的数字

题目描述:

参考代码:
class Solution {
public:
    int singleNumber(vector<int>& nums) {
        int ret = 0;
        for(auto n :nums)
        {
            ret ^= n;

        }
        return ret;
    }
};

260.只出现一次的数字|||

题目链接:只出现一次的数字|||

题目描述:

参考代码:
class Solution {
public:
    vector<int> singleNumber(vector<int>& nums) {
        unsigned int s = 0;
        
        for(auto n : nums)
        {
            s ^= n;
        }
        int low = s & -s;//取出最右侧的1
        int a = 0,b = 0;
        for(auto n : nums)
        {
            if((low & n) == low)
            {
                a ^= n;
            }
            else
            {
                b ^= n;
            }
        }
        return vector<int>{a,b};
    }
};

二、判断字符是否唯一

题目链接:判断字符是否唯一

题目描述:

解法一:我们可以使用哈希表

class Solution1 {
public:
    bool isUnique(string astr) {
        unordered_set<int>hash;
        for (auto ch : astr)
        {
            if (hash.count(ch)) return false;
            hash.emplace(ch);
        }
        return true;
    }
};

解法二:位图

我们用0表示没出现过,1表示出现过

可以利用鸽巢原理来进行优化,鸽巢原理已经在双指针那里讲过了这里就不过多赘述,一共有26个字母如果字符串的长度超过则肯定有重复字符,如果字符串的长度大于26那么直接返回false

参考代码:

class Solution {
public:
    bool isUnique(string astr) {
        //利用鸽巢原理来做的优化
        if (astr.size() > 26) return false;
        int bitMap = 0;
        for (auto ch : astr)
        {
            int i = ch - 'a';
            //判断字符是否已经出现过
            if ((bitMap >> i) & 1 == 1) return false;
            //把当前字符加入位图中
            bitMap |= 1 << i;
        }
        return true;
    }
};

三、丢失的数字

题目链接:丢失的数字

题目描述:

解法一:哈希表

创建一个哈希表然后遍历数组,0出现标记一下,1出现标记一下,3出现标记一下....

解法二:高斯求和

(首项 + 尾项) * 项数 / 2这样就算出了0~5的和然后我们再减去数组里面所有数之和这样就得出来了

参考代码:

class Solution1 {
public:
    int missingNumber(vector<int>& nums) {
        int n = nums.size();
        int sum = n * (n + 1) / 2;
        int ret = 0;
        for (auto n : nums)
        {
            ret += n;
        }
        return sum - ret;
    }
};

解法三:位运算(异或运算的运算律)

参考代码:

class Solution {
public:
    int missingNumber(vector<int>& nums)
    {
        int ret = 0;
        for (int i = 0; i <= nums.size(); i++) ret ^= i;
        for (auto n : nums) ret ^= n;
        return ret;
    }
};

四、两整数之和

题目链接:两整数之和

题目描述:

在笔试中我们不讲武德直接return a + b;

解法:位运算(异或运算-无进位相加)

13+28=41

a:       0 0 1 1 0 1

b:       0 1 1 1 0 0

——————————

a^b:   0 1 0 0 0 1(a)   无进位

进位(a & b)<<1   0 1 1 0 0 0         我们进位是往前进位所以这里我们右移一位

我们继续重复如上操作,先无进位相加再进位

a:               0 1 0 0 0 1

b:               0 1 1 0 0 0 

a^b:           0 0 1 0 0 1  无进位

(a & b) <<1  1 0 0 0 0 0  进位

 a:              0 0 1 0 0 1 

 b :             1 0 0 0 0 0 

a^b:           1 0 1 0 0 1  无进位    41

(a & b) <<1   0 0 0 0 0 0 进位

进位变成0就结束了,最后的无进位相加就是我们的最终结果

参考代码:

class Solution {
public:
    int getSum(int a, int b) 
    {
        while(b != 0)
        {
            int x = a ^ b;//无进位相加
            unsigned carry = (unsigned)(a & b) <<1;//算出进位
            a = x;
            b = carry;
        }
        return a;
    }
};

五、只出现一次的数字|| 

题目链接:只出现一次的数字|| 

题目描述:

设要找的数位 ret ,由于整个数组中,需要找的元素只出现了⼀次,其余的数都出现三次,因此我们可以根据所有数的某⼀个⽐特位的总和 %3 的结果,快速定位到 ret 的⼀个⽐特位上的值是 0 还是 1 。 这样,我们通过 ret 的每⼀个⽐特位上的值,就可以将 ret 给还原出来。

 参考代码:

class Solution {
public:
    int singleNumber(vector<int>& nums) 
    {
        int ret = 0;
        for(int i = 0;i < 32;i++)
        {
            int sum = 0;
            for(int x : nums)
                if(((x>>i) & 1) == 1)
                    sum++;
            sum %= 3;
            if(sum == 1) ret |= 1<<i;
        }
        return ret;
    }
    
};

六、消失的两个数字

题目链接:消失的两个数字

题目描述:

这道题其实是丢失的数字+只出现一次的数字|||融合一起,本题的算法原理就是用到了这两道题的一个算法原理。

nums中消失了两个数字,1~N这堆数中假设a和b是消失的两个数字,nums这一堆数和1~N这一堆数异或,其他的数出现了2次a和b出现了一次,那么其实就是a ^ b

解法:位运算

1.将所有的数异或在一起,tmp

tmp = a ^ b

2.找到tmp中,比特位上为1的那一位

a^b的结果肯定不为0因为他们是不同的数,所以它们的比特位上肯定有一位是1,a和b的第x位上肯定是不同的

3.根据x位的不同,划分成两类异或

我们可以把x位是0的分为一类,x位上是1的分一类,然后对两组数据分别进行异或。

参考代码:

class Solution {
public:
    vector<int> missingTwo(vector<int>& nums) 
    {
        //1.把所有数异或起来
        int tmp = 0;
        for(auto n : nums) tmp ^= n;
        for(int i = 1;i<=nums.size()+2;i++) tmp ^= i;
        int diff = 0;
        //找出a,b比特位不同的那一位
        while(1)
        {
            if(((tmp >>diff) & 1) == 1) break;
            else diff++;
        }
        //3.根据diff位的不同,将所有数划分两类来异或
        int a = 0,b = 0;
        for(auto n : nums)
            if(((n >> diff) & 1) == 1) b ^= n;
            else a ^= n;
        for(int i = 1;i<=nums.size()+2;i++)
        {
            if(((i >> diff) & 1) == 1) b ^= i;
            else a ^= i;
        }
        return {a,b};
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2292013.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Cadence仿真技巧学习笔记】求解65nm库晶体管参数un, e0, Cox

在设计放大器的第一步就是确定好晶体管参数和直流工作点的选取。通过阅读文献&#xff0c;我了解到L波段低噪声放大器的mos器件最优宽度计算公式为 W o p t . p 3 2 1 ω L C o x R s Q s p W_{opt.p}\frac{3}{2}\frac{1}{\omega LC_{ox}R_{s}Q_{sp}} Wopt.p​23​ωLCox​Rs…

Docker入门篇(Docker基础概念与Linux安装教程)

目录 一、什么是Docker、有什么作用 二、Docker与虚拟机(对比) 三、Docker基础概念 四、CentOS安装Docker 一、从零认识Docker、有什么作用 1.项目部署可能的问题&#xff1a; 大型项目组件较多&#xff0c;运行环境也较为复杂&#xff0c;部署时会碰到一些问题&#xff1…

开源智慧园区管理系统对比其他十种管理软件的优势与应用前景分析

内容概要 在当今数字化快速发展的时代&#xff0c;园区管理软件的选择显得尤为重要。而开源智慧园区管理系统凭借其独特的优势&#xff0c;逐渐成为用户的新宠。与传统管理软件相比&#xff0c;它不仅灵活性高&#xff0c;而且具有更强的可定制性&#xff0c;让各类园区&#…

【C++】P5734 【深基6.例6】文字处理软件

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;题目描述&#x1f4af;题目描述输入格式输出格式示例输入与输出输入&#xff1a;输出&#xff1a; &#x1f4af;我的做法操作1&#xff1a;在文档末尾插入字符串操作2&…

CSS核心

CSS的引入方式 内部样式表是在 html 页面内部写一个 style 标签&#xff0c;在标签内部编写 CSS 代码控制整个 HTML 页面的样式。<style> 标签理论上可以放在 HTML 文档的任何地方&#xff0c;但一般会放在文档的 <head> 标签中。 <style> div { color: r…

013-51单片机红外遥控器模拟控制空调,自动制冷制热定时开关

主要功能是通过红外遥控器模拟控制空调&#xff0c;可以实现根据环境温度制冷和制热&#xff0c;能够通过遥控器设定温度&#xff0c;可以定时开关空调。 1.硬件介绍 硬件是我自己设计的一个通用的51单片机开发平台&#xff0c;可以根据需要自行焊接模块&#xff0c;这是用立创…

CMake项目编译与开源项目目录结构

Cmake 使用简单方便&#xff0c;可以跨平台构建项目编译环境&#xff0c;尤其比直接写makefile简单&#xff0c;可以通过简单的Cmake生成负责的Makefile文件。 如果没有使用cmake进行编译&#xff0c;需要如下命令&#xff1a;&#xff08;以muduo库echo服务器为例&#xff09;…

OPENPPP2 —— VMUX_NET 多路复用原理剖析

在阅读本文之前&#xff0c;必先了解以下几个概念&#xff1a; 1、MUX&#xff08;Multiplexer&#xff09;&#xff1a;合并多个信号到单一通道。 2、DEMUX&#xff08;Demultiplexer&#xff09;&#xff1a;从单一通道分离出多个信号。 3、单一通道&#xff0c;可汇聚多个…

语言月赛 202412【正在联系教练退赛】题解(AC)

》》》点我查看「视频」详解》》》 [语言月赛 202412] 正在联系教练退赛 题目背景 在本题中&#xff0c;我们称一个字符串 y y y 是一个字符串 x x x 的子串&#xff0c;当且仅当从 x x x 的开头和结尾删去若干个&#xff08;可以为 0 0 0 个&#xff09;字符后剩余的字…

【数据结构】_链表经典算法OJ:复杂链表的复制

目录 1. 题目链接及描述 2. 解题思路 3. 程序 1. 题目链接及描述 题目链接&#xff1a;138. 随机链表的复制 - 力扣&#xff08;LeetCode&#xff09; 题目描述&#xff1a; 给你一个长度为 n 的链表&#xff0c;每个节点包含一个额外增加的随机指针 random &#xff0c;…

python的pre-commit库的使用

在软件开发过程中&#xff0c;保持代码的一致性和高质量是非常重要的。pre-commit 是一个强大的工具&#xff0c;它可以帮助我们在提交代码到版本控制系统&#xff08;如 Git&#xff09;之前自动运行一系列的代码检查和格式化操作。通过这种方式&#xff0c;我们可以确保每次提…

【C语言入门】解锁核心关键字的终极奥秘与实战应用(三)

目录 一、auto 1.1. 作用 1.2. 特性 1.3. 代码示例 二、register 2.1. 作用 2.2. 特性 2.3. 代码示例 三、static 3.1. 修饰局部变量 3.2. 修饰全局变量 3.3. 修饰函数 四、extern 4.1. 作用 4.2. 特性 4.3. 代码示例 五、volatile 5.1. 作用 5.2. 代码示例…

音标-- 02-- 重音 音节 变音

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 国际音标1.重音2.音节3.变音 国际音标 1.重音 2.音节 3.变音

[STM32 标准库]EXTI应用场景 功能框图 寄存器

一、EXTI 外部中断在嵌入式系统中有广泛的应用场景&#xff0c;如按钮开关控制&#xff0c;传感器触发&#xff0c;通信接口中断等。其原理都差不多&#xff0c;STM32会对外部中断引脚的边沿进行检测&#xff0c;若检测到相应的边沿会触发中断&#xff0c;在中断中做出相应的处…

C语言练习【互斥锁、信号量线程同步、条件变量实现生产者消费者模型】

练习1 请使用互斥锁 和 信号量分别实现5个线程之间的同步 互斥锁实现同步 #include <stdio.h> #include <string.h> #include <unistd.h> #include <stdlib.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h>…

w190工作流程管理系统设计与实现

&#x1f64a;作者简介&#xff1a;多年一线开发工作经验&#xff0c;原创团队&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的网站项目。 代码可以查看文章末尾⬇️联系方式获取&#xff0c;记得注明来意哦~&#x1f339;赠送计算机毕业设计600个选题excel文…

linux下ollama更换模型路径

Linux下更换Ollama模型下载路径指南   在使用Ollama进行AI模型管理时&#xff0c;有时需要根据实际需求更改模型文件的存储路径。本文将详细介绍如何在Linux系统中更改Ollama模型的下载路径。 一、关闭Ollama服务   在更改模型路径之前&#xff0c;需要先停止Ollama服务。…

编程题-电话号码的字母组合(中等)

题目&#xff1a; 给定一个仅包含数字 2-9 的字符串&#xff0c;返回所有它能表示的字母组合。答案可以按 任意顺序 返回。 给出数字到字母的映射如下&#xff08;与电话按键相同&#xff09;。注意 1 不对应任何字母。 解法一&#xff08;哈希表动态添加&#xff09;&#x…

浅谈《图解HTTP》

感悟 滑至尾页的那一刻&#xff0c;内心突兀的涌来一阵畅快的感觉。如果说从前对互联网只是懵懵懂懂&#xff0c;但此刻却觉得她是如此清晰而可爱的呈现在哪里。 介绍中说&#xff0c;《图解HTTP》适合作为第一本网络协议书。确实&#xff0c;它就像一座桥梁&#xff0c;连接…

架构知识整理与思考(其四)

书接上回 建议&#xff0c;没有看过上一章的可以看一下&#xff0c;上一章“架构知识整理与思考&#xff08;其二&#xff09;” 感觉这都成链表了。 三生万物 软件架构 终于&#xff0c;我们进入了具体的软件架构讨论中。 软件架构是什么&#xff1f;相关定义如下&#xf…