算法专题:位运算

news2025/1/23 22:27:58

目录

常见位运算总结

位运算相关算法题

1. 只出现一次的数字

2. 只出现一次的数字(|||)

3. 两整数之和

4. 只出现一次的数字(||)


常见位运算总结

        在开始刷位运算这个类型的题目前,我想先带着大家学习一下一些常见的基础运算操作,我们后面刷题时对题目的处理都是基于这些基本运算。

1. 基础位运算

        想必大家对位运算符都是非常熟悉的,在这里,我为大家讲讲我对几个按位操作符的理解

        &:有0就是0

        | :有1就是1

        ^ :相同为0,相反为1 (也可以视作为无进位相加)

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

        (n >> x) & 1

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

        n |=(1 << x)

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

        n &= (~(1 << x))(~就是取反)

5. 位图的思想

        位图的本质就是个哈希表,我们先前学习过的哈希表在大部分情况下都是数组形式的

使用一个变量的比特位来记录信息,对哈希表的增删查改,就相当于是对比特位的增删查改,这就意味着,我们前面讲到的几种常见的位运算操作都是非常重要的,能帮助我们更好地操作位图

6. 提取一个数n最右侧的1

        这个描述听起来有些抽象,我为大家举个例子:

        这就是提取最右侧的1

        操作方式:n & -n

        -n是由n按位取反再+1得到的:

        通过这个例子我们发现了:n与-n的关系是,在最右侧的1的左侧,n与-n完全相反;在最右侧的1的右侧,n与-n完全相同。则左侧区域两者&的结果必定为0,右侧区域&的结果不变,于是我们实现了把最右侧1提取出来的操作。

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

        n & (n - 1)

        众所周知,减法是存在借位的,n - 1的本质就是,将最右侧的1右侧的区域全部取反

8. 异或运算的运算律

        现在我们已经学习完了常见的操作,接下来开始刷题吧!大家可以先试着做以下几道题练练手,根据我们的总结可以直接处理:

        191. 位1的个数 - 力扣(LeetCode)

        

        461. 汉明距离 - 力扣(LeetCode)

        接下来我们来另外刷下这几道题。希望大家在看讲解前,先自己动手编写代码,实在没有思路时再去看答案,这样才能实际提高水平~

位运算相关算法题

1. 只出现一次的数字

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

题目解析:

        这道题目如果我们没有学习过位运算,可能只能想到创建一个哈希表,通过哈希表来找到只出现一次的元素。但事实上,根据我们在常见位运算总结中的第八点:任何书亦或0的结果为这个数本身、两数相同时为0、异或运算满足结合律,我们可以得出一个结论:在数组中出现了两次的元素可以结合为0,因为只出现了一次的元素只有一个,所以我们对整个数组求亦或和,得到的结果就是只出现一次的元素!

代码:

class Solution {
public:
    int singleNumber(vector<int>& nums) 
    {
        int k = 0;
        for(auto num : nums)
        {
            k ^= num;
        }
        return k;
    }
};

2. 只出现一次的数字(|||)

260. 只出现一次的数字 III - 力扣(LeetCode)

题目解析:

        本题如果使用哈希表,可以非常轻松的解决,但由于面试中,面试官可能会问我们有没有其他的解法,所以我们这里试着用位运算来处理。本题与上一题的区别在于,仅出现一次的元素有两个,我们就不能简单地求整个数组的亦或和,而是要试着把这两个数字分离开。

        我们依然先求出整个数组的亦或和sum,因为两个数字都是只出现一次的,所以sum的比特位必定有一位是1,因为sum可以被视为是两个数字亦或得到的,所以sum的每个为1的比特位对应的都是两个数字相异的比特位!这正好符合我们要将这唯二两个只出现一次的数字分离开的需求,所以接下来我们只需要提取出sum中的一个1,与整个数组相与就能把两个数字分离了,在常见位运算总结中,我们学到了提取最右侧1的方法:n&-n,所以我们可以轻松提取出sum的最右侧的1,根据这个作为条件进行判断即可。

        大家可能会奇怪,我只提到了分离这两个只出现一次的数字,那出现两次的呢?当然是因为对于出现两次的数字,在条件判断时,两个都会被分到其中的一组,则我们接下来对这两组元素分别求亦或和,就能分别得到两个只出现一次的数字了。

class Solution {
public:
    vector<int> singleNumber(vector<int>& nums) 
    {
        int orsum = 0;
        for(auto num : nums)
        {
            orsum ^= num;
        }
        int jud = (orsum == INT_MIN) ? orsum : orsum & (-orsum);
        int type1 = 0, type2 = 0;
        for(auto num : nums)
        {
            if(num & jud)
            {
                type1 ^= num;
            }
            else
            {
                type2 ^= num;
            }
        }
        return {type1, type2};
    }
};

3. 两整数之和

371. 两整数之和 - 力扣(LeetCode)

题目解析:

        本题乍一看无从下手,但是其实可以用我们总结的位运算操作来实现,还记得么,亦或运算实际上就是无进位相加,那么我们只要把两个数的亦或和加上他的进位就行了。

        至于进位,运算时仅当两个比特位都为1时才产生进位,是不是让我们联想到了与运算,仅当两个比特位都为1时才为1,由于进位的结果要加到更高的比特位上,所以我们将两个数的按位与之和左移一位,得到的就是本次相加的进位,接下来把进位、无进位相加结果相加即可。

        可是这不就又涉及到相加了吗?所以我们要重复这个流程,运算中不再有进位为止,这样就实现了不适用+、-进行两个整数的相加。

        还有一个细节问题,有符号整数在内存中是以补码的形式存储的,对于补码的运算规则是:

对于补码的移位运算,如果他的最高位与符号位是不一样的,左移的时候就会出现问题,举个例子:

此时我们发现,这个数左移时,符号位发生了变化,也就是说我们的左移操作是有问题的,为了避免这种情况,我们使用unsigned int来代替int进行储存变量

class Solution {
public:
    int getSum(int a, int b) 
    {
        while(b)
        {
            int x = a ^ b;
            unsigned int carry = (unsigned int)(a & b) << 1;
            a = x;
            b = carry;
        }
        return a;
    }
};

4. 只出现一次的数字(||)

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

题目解析:

        本题和前面两道只出现一次的数字一样,都可以用哈希表来非常轻松的解决,但是如果要求不使用额外空间来实现的话,就要用到位图了。

        依据题意,给定的元素可以被分为两类:出现一次的元素、出现三次的元素,对于整型数的32个比特位的每一位而言,可能的取值为0、1,则如果我们求这两类元素某一个比特位的和,由于第二类元素都出现了三次,假设nums范围为[0, n],则这类元素的和一定等于3 * m(m <= n).

        至于只出现一次的元素,在该比特位可能为0、1,所以对所有元素来说,该比特位的和可能值为:3 * m、3 * m + 1,对这个结果取模3,得到的可能结果为0、1,这正是只出现一次的元素的这一比特位,所以我们可以依法炮制,求出所有比特位。

        至于判断某个比特位是否为1、将某个比特位修改为1的方式,我们都在常见位运算总结部分提到了,所以直接上手写代码:

class Solution {
public:
    int singleNumber(vector<int>& nums) 
    {
        int ret = 0;
        for(int i = 0; i < 32; i++)
        {
            int sum = 0;
            for(auto &num : nums)
            {
                if((num >> i) & 1) sum++;
            }
            // 如果比特位和模3余1,将返回值的该比特位修改为1
            if(sum % 3 == 1)
                ret |= (1 << i);
        }
        return ret;
    }
};

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

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

相关文章

Abp框架,EF 生成迁移文件时,自动添加表和字段注释内容

在使用 abp 框架&#xff0c;或者ef 的时候都会遇到一个问题&#xff0c;就是建实体后要将实体描述生成到数据库中&#xff0c;就需要手动去添加 [Comment("注释内容")] 注解&#xff0c;这样相当于手动写两次注释&#xff08;即使你是 Ctrl C&#xff09;&#x…

现在做电商迟吗?那是你不知道今年黑马,视频号小店重磅来袭

大家好&#xff0c;我是电商笨笨熊 24年想做电商&#xff0c;还能不能做&#xff1f; 当然可以。 电商是一个长期的市场&#xff0c;只要用户有需求&#xff0c;那么电商就会一直存在&#xff1b; 尤其是近几年来无货源模式爆火&#xff0c;对于我们商家来说这种无需自备货…

软件安全测试可以检测软件哪些安全问题?

软件安全测试是一种旨在发现和评估软件应用程序中的安全漏洞和隐患的测试方法。通过安全测试&#xff0c;可以发现并修复潜在的安全问题&#xff0c;从而提高软件应用程序的可靠性和安全性。下面将介绍软件安全测试可以检测到的几种主要安全问题。 身份验证漏洞&#xff1a;身份…

【教学类-55-02】20240512图层顺序挑战(四格长条纸加黑色边框、4*4、7张 、43200张去掉非7色有23040张,去掉重复样式有几种?)

作品展示 背景需求&#xff1a; 之前的代码吗存在几个问题&#xff0c;最大的问题是不能生成“”长条黑边框”” 【教学类-55-01】20240511图层顺序挑战&#xff08;四格长条纸&#xff09;&#xff08;4*4&#xff09;和“手工纸自制参考图”-CSDN博客文章浏览阅读485次&…

黑盒测试中的边界值分析

黑盒测试是一种基于需求和规格的测试方法&#xff0c;它主要关注软件系统输出的正确性和完整性&#xff0c;而不考虑内部代码的实现方式。在黑盒测试中&#xff0c;边界值分析是一种重要的测试技术&#xff0c;它可以帮助测试人员有效地发现输入和输出的问题。本文将从什么是边…

nacos在没有指定数据源的情况下默认使用什么数据库?

在没有特别指定数据源的情况下&#xff0c;Nacos 默认使用内嵌的数据库 Derby 来存储其数据。Derby 是一个轻量级的、基于 Java 的数据库管理系统&#xff0c;适合于开发和测试环境&#xff0c;因为它简单易部署且无需额外的数据库服务器。然而&#xff0c;对于生产环境&#x…

JSP ssm 房屋中介管理myeclipse开发mysql数据库springMVC模式java编程计算机网页设计

一、源码特点 JSP ssm 房屋中介管理系统是一套完善的web设计系统&#xff08;系统采用SSM框架进行设计开发&#xff0c;springspringMVCmybatis&#xff09;&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采…

【C语言/数据结构】栈:从概念到两种存储结构的实现

目录 一、栈的概念 二、栈的两种实现方式 1.顺序表实现栈 2.链表实现栈 三、栈的顺序存储结构及其实现 1.栈的声明 2.栈的初始化 3.栈的销毁 4.栈的压栈 5.栈的弹栈 6.栈的判空 7.返回栈顶元素 8.返回栈的长度 四、栈的链式存储结构及其实现 1.栈的声明 2.栈的…

GBase 8s 数据库集群切换及恢复

GBase 8s 数据库切换分为自动切换、由CM控制的按FOC规则的切换、手工切换。 自动切换 全自动切换用于HAC集群中&#xff0c;由于集群只有两个节点&#xff0c;数据库相互之前进行状态检查&#xff0c;发现异常时&#xff0c;能按DRAUTO的配置方式进行自动切换。 在HAC集群中&…

ES扩缩容

ES扩容 1.1 页面扩容ES1 1.2 拷贝插件及ssl文件 JSON [ec_admin@kde-offline3 ~]$ sudo rsync -avP /usr/kde_ec/2.3.6.6-1/elasticsearch1/plugins/* kde-offline6:/usr/kde_ec/2.3.6.6-1/elasticsearch1/plugins/ ;echo $? [ec_admin@kde-offline3 ~]$ sudo rsync -avP /us…

SQL Server中怎么排查死锁问题

一、背景 我们在UAT环境压测的时候&#xff0c;遇到了如下的死锁异常。 Caused by: com.microsoft.sqlserver.jdbc.SQLServerException: Transaction (Process ID 82) was deadlocked on lock resources with another process and has been chosen as the deadlock victim. Re…

Windows下安装Node.js、npm和electronic,并运行一个Hello, World!脚本程序

20240510 By wdhuag 目录 简介&#xff1a; 参考&#xff1a; 安装Node.js 安装npm 配置npm&#xff1a; 修改包存放目录和缓存目录 切换镜像源 使用 nrm 切换镜像源 安装Electron 运行一个Hello, World!脚本程序 安装Yarn JavaScript 指南 简介&#xff1a; Nod…

【Spring】GoF 之代理模式

一、代理模式 在 Java 程序中的代理模式的作用&#xff1a; 当一个对象需要受到保护的时候&#xff0c;可以考虑使用代理对象去完成某个行为 需要给某个对象的功能进行功能增强的时候&#xff0c;可以考虑找一个代理进行增强 A 对象无法和 B 对象直接交互时&#xff0c;也可以…

MySQL5.7压缩包安装图文教程

一、下载 https://dev.mysql.com/downloads/mysql/ 选择5.7版本 二、解压 下载完成后解压&#xff0c;解压后如下&#xff08;zip是免安装的&#xff0c;解压后配置成功即可使用&#xff09; 注意&#xff1a;只有5.6以前的版本才有在线安装&#xff08;install msi&#xf…

徐孝雅:一位清华毕业温州人的科技报国之路

图 徐孝雅 偶然的一次聚会,朋友讲述一位科技企业家的传奇故事,出于记者的敏感,我对这位企业家产生了好奇。 2024年4月,我们如约见到这位企业家,他叫徐孝雅。他给我的印象极其深刻,中等身材,儒雅大方,气度不凡,眼神展露坚定和自信,浑身散发神采和活力。寒暄之后,似乎一见如故…

armbian 安装libreoffice 转换word为PDF

安装libreoffice sudo apt-get install libreoffice安装JVM sudo apt-get install default-jre #验证 java -version尝试转换&#xff1a; libreoffice --convert-to pdf /root/printFiles/f.docx发现问题乱码 从Windows 拷贝字体到debian上&#xff0c;windows字体路径是&a…

第十二讲:指针(4)

第十二讲&#xff1a;指针&#xff08;4&#xff09; 1.回调函数1.1什么是回调函数1.2深入理解并使用回调函数1.2.1简单写法1.2.2优化 2.qsort函数详解2.1函数简单介绍2.3qsort函数使用举例2.3.1qsort函数排序整形数据2.3.2qsort函数排序结构数据 3.qsort函数的模拟实现3.1冒泡…

网络实验新境界,PNETLab模拟器部署指南

在网络工程领域&#xff0c;拥有一个可靠的网络实验平台至关重要。PNETLab模拟器是一款功能强大的网络仿真工具&#xff0c;它支持包括华为、华三、锐捷、思科在内的多种设备&#xff0c;并且以开源免费的形式提供&#xff0c;这使得它在业界备受青睐。 软件介绍 PNETLab&am…

ThingsBoard版本控制配合Gitee实现版本控制

1、概述 2、架构 3、导出设置 4、仓库 5、同步策略 6、扩展 7、案例 7.1、首先需要在Giitee上创建对应同步到仓库地址 ​7.2、giit仓库只能在租户层面进行配置 7.3、 配置完成后&#xff1a;检查访问权限。显示已成功验证仓库访问&#xff01;表示配置成功 7.4、添加设…

【代码随想录】【动态规划】背包问题 - 完全背包

完全背包 模板&#xff1a;完全背包问题 问题描述 完全背包问题与01背包问题唯一的区别在于&#xff1a; 在01背包中&#xff1a;每个物品只有一个&#xff0c;要么放入背包&#xff0c;要么不放入背包在完全背包中&#xff1a;每个物品有无限多个&#xff0c;可以不放入背…