算法专题:位运算

目录

常见位运算总结

位运算相关算法题

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,将返回值的该比特位修改为1if(sum % 3 == 1)ret |= (1 << i);}return ret;}
};

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

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

相关文章

2024年成都市企业技术中心认定申报条件要求、评价标准和时间

一、2024年成都市企业技术中心认定 &#xff08;一&#xff09;申报条件 1&#xff0e;在成都市行政区域内注册&#xff0c;具有独立法人资格。 2&#xff0e;已建立企业技术中心并正常运行1年以上。 3&#xff0e;有较强的经济、技术实力和较好的经济效益&#xff0c;在同…

Funkey游戏机新作,基于全志T113的全新版本

不同于配置高端、性能强劲的Windows、安卓掌机&#xff0c;有一部分的爱好者往往对拥有复古外形的开源掌机更加感兴趣。作为开源掌机的热门产品&#xff0c;小巧便携的FunKeys掌机是各位开源爱好者争相复刻的对象。因热爱开源掌机DIY而聚集的“双核掌机开发组”开发者团队&…

【python量化交易】qteasy使用教程05——创建第一个自定义交易策略

创建第一个自定义交易策略 使用qteasy创建自定义交易策略开始前的准备工作本节的目标自定义策略的实现方法使用 qteasy 的 Strategy 策略类三种不同的自定义策略基类定义一个双均线择时交易策略定义策略运行时机定义策略需要的数据自定义交易策略的实现&#xff1a;realize()获…

OpenGL入门第四步:摄像机视角变换与交互

OpenGL入门第一步:创建窗口、重写虚函数-CSDN博客 OpenGL入门第二步:颜色、纹理设置(解析)-CSDN博客 OpenGL入门第三步:矩阵变换、坐标系统-CSDN博客 目录 函数解析 具体代码 函数解析 相机视角变换需要与鼠标键盘进行交互,需要重写鼠标和键盘响应函数。 初始化 …

【Java】获取近六个月的年月

数据库里面存储的字段类型就是varchar&#xff0c;数据格式就是类似2024-12这样的年月格式。 目标&#xff1a; 以当前月份为标准&#xff0c;向前获取近6个月的年月&#xff08;year_month&#xff09;形成列表 // 获取近6个月的年月列表List<String> recentMonths ge…

java项目之相亲网站的设计与实现源码(springboot+mysql+vue)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于springboot的相亲网站的设计与实现。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 相亲网站的设计与实…

Unable to locate the .NET SDK

问题描述&#xff1a; vs2019 加载项目时&#xff0c;提示如下&#xff1a; Unable to locate the .NET SDK as specified by global.json, please check that the specified version is installed. 项目中没有globan找al.json 文件 先使用&#xff1a; dotnet --list-sdks 命…

论文研读 Disentangled Information Bottleneck

解耦信息瓶颈 摘要&#xff1a; 信息瓶颈方法是一种从源随机变量中提取与预测目标随机变量相关的信息的技术&#xff0c;通常通过优化平衡压缩和预测项的IB拉格朗日乘子f来实现&#xff0c;然而拉格朗日乘子很难优化&#xff0c;需要多次实验来调整拉格朗日乘子的值&#xff0c…

mybatis 跨库查询 mysql

跨库&#xff0c;表关联的查询&#xff0c;实现起来很简单&#xff1a; select a.uid from ucenter.user a , database user_profile b where a.uid b.uid;只要在表的前边加上库名即可。 这个是我项目中xml 中的一个例子&#xff0c;项目采用的是springmvc,持久层框架就是my…

Screeps工程化之配置化

目录 前言一、抽取配置项二、读取配置项 前言 Screeps中所有代码都会在一个tick&#xff08;游戏内的世间&#xff09;内执行完成&#xff0c;想要做到代码的高度复用&#xff0c;和隔离各个房间creep的行为就需要将部分代码进行配置化&#xff0c;本文仅为作者本人的游戏思路…

反了!美国假冒邮政服务钓鱼网站访问量竟然超过正规官网

美国邮政是美国主要的包裹信件投递机构之一&#xff0c;长期以来该单位都是网络钓鱼和诈骗的针对目标。对美国公民来说&#xff0c;在假期通常都会收到声称来自美国邮政的诈骗。美国邮政甚至单独建设的网页提醒消费者警惕诈骗信息&#xff1a; 专用提醒网页 Akamai 的研究人员…

ABB机器人转角路径故障报警消除方法

ABB机器人在现场调试时&#xff0c;有时候会出现以下报警&#xff1a;“转角路径故障”的错误。 但这个报错不影响机器人的使用。也可以在指令中设置将其屏蔽。 1、打开一个例行程序&#xff0c;在Settings指令下添加CornerPathWarning设置语句&#xff1b; 2、将CornerPathWa…

使用Pandas对Data列进行基于顺序的分组排列

目录 一、引言 二、Pandas库简介 三、按照数据列中元素出现的先后顺序进行分组排列 四、案例分析 五、技术细节探讨与扩展应用 1. 技术细节 2. 扩展应用 3. 示例代码&#xff1a;用户行为分析 4. 进阶应用&#xff1a;分组后的聚合操作 5. 分组后的数据筛选 6. 分组…

信息系统安全与对抗-网络侦查技术与网络扫描技术(期末复习简答题)

1、网络拓扑结构在网络攻击中的作用 查明目标网络的拓扑结构&#xff0c;有利于找到目标网络的关键节点&#xff0c;从而提高攻击效率&#xff0c;达到最大攻击效果。 2、网络侦查在网络攻击中的作用 识别潜在目标系统&#xff0c;确认目标系统适合哪种类型的攻击。 3、百度…

视频号小店究竟有什么秘密,值得商家疯狂入驻,商家必看!

大家好&#xff0c;我是电商花花。 我们都知道视频号和抖音本身都是一个短视频平台&#xff0c;但是随着直播电商的发展&#xff0c;背后的流量推动逐步显露出强大的红利市场和变现机会。 视频号小店流量大和赚钱之外&#xff0c;还非常适合普通人创业。 这也使得越来越多的…

[机器学习-03] Scikit-Learn机器学习工具包学习指南:主要功能与用法解析

&#x1f3a9; 欢迎来到技术探索的奇幻世界&#x1f468;‍&#x1f4bb; &#x1f4dc; 个人主页&#xff1a;一伦明悦-CSDN博客 ✍&#x1f3fb; 作者简介&#xff1a; C软件开发、Python机器学习爱好者 &#x1f5e3;️ 互动与支持&#xff1a;&#x1f4ac;评论 &…

robobrowser,一个有趣的 Python 库!

更多资料获取 &#x1f4da; 个人网站&#xff1a;ipengtao.com 大家好&#xff0c;今天为大家分享一个有趣的 Python 库 - robobrowser。 Github地址&#xff1a;https://github.com/jmcarp/robobrowser 在网络爬虫和自动化领域&#xff0c;Python开发者拥有众多强大的工具&…

Elasticsearch查看集群信息,设置ES密码,Kibana部署

Elasticsearch查看集群信息&#xff0c;设置ES密码&#xff0c;Kibana部署 查看集群信息查看节点信息查看集群健康状态查看分片信息查看其他集群信息 Kibana部署安装设置ES密码 查看集群信息 查看节点信息 curl http://127.0.0.1:9200/_cat/nodes?v 参数说明&#xff1a; ip…

YOLOv8火焰与烟雾智能检测系统

项目概述&#xff1a; 本项目旨在开发一款高效、实时的火焰与烟雾检测系统&#xff0c;利用先进的深度学习技术——YOLOv8&#xff0c;为安全监控领域提供智能化解决方案。系统不仅能够准确识别视频流或静态图像中的火焰与烟雾&#xff0c;还配备了用户友好的图形界面&#xff…

AI 绘画神器 Fooocus 2.3.1 汉化教程(中文界面/汉化包下载/持续更新最新版本...)

本文收录于《AI绘画从入门到精通》专栏&#xff0c;专栏总目录&#xff1a;点这里&#xff0c;订阅后可阅读专栏内所有文章。 大家好&#xff0c;我是水滴~~ Fooocus 是一款功能强大的 AI 绘画神器&#xff0c;它能够帮助我们以更高效、更创意的方式进行绘画创作。本教程将详细…