算法——位运算(一篇搞定)

本专栏为大家分享本人学习算法遇到的不同类型的题目以及解析!''

此篇文章给大家分享一些关于位运算算法的题目,在开篇还讲述了常见位运算的公式以及题目,如果对您有帮助,麻烦点个关注,如有错误,请您指出!

1.常见位运算总结(包含5道题目)

1.1基础位运算
运算符操作
<<左移
>>右移
~取反
&对应二进制位有0就是0
|对应二进制位有1就是1
^对应二进制位,相同为0相异为1 / 不进位相加

对于^运算的不进位相加解释:假设为

0 1 0

0 1 1

那么不进位相加就是 0 0 1,这个理解在我们后面是有用的

1.2给一个数n,确定他的二进制位中的第x为是0还是1

利用&(对应二进制位有0就是0)运算即可,将 n >> x后,他的第x为就在最右边,那么此时& 一个1,那么就只会得到1 或 0,如果是1,那么这个位上是1,反之则是0

即 (n>> x) &1

1.3将一个数n的二进制表示的第x位修改为1

方法: n | (1 << x) 即可:

这样,能保证其他位的不变,将对应x位改为1

1.4将一个数n的二进制表示的第x位修改为0

方法: 要想将某一位修改为0,那么就让这一位 & 0即可,但是怎么得到仅是这一位为0的数字呢??

我们将1 << x后取反即可

这样,通过 n & (~(1 << x))就能得到所求

1.5位图的思想

哈希表实际上大多数情况下是个数组,通过数组的内容来存储数据

实际上二进制位同样有这个效果:

通过二进制位上是0还是1来存储信息,而我们上面的1.2 ~ 1.4的方法主要是为了以后操作位图做好准备

1.6提取一个数n二进制表示中最右侧的1

方法: n & -n

-n操作实际上就是将原数取反后 +1 的操作,

这样我们会发现,n与-n相比,原数最右侧的1的左边变成原来相反,右侧则不变,还是0

那么我们将 n & -n后,就能提取出最右侧的1

1.7去掉一个数n二进制表示中最右侧的1

方法: n & (n-1)

实际上n-1的操作就是从右往左进行借位的操作,因为在遇到最右侧的1之前,其余位都是0,是不够-1的,那么就要借位,直到遇见1

那么n-1后就得到与原来的n相比:最右侧的1(包括这个1)的右侧都是原来的相反数,而左侧的不变

那么我们将n & n-1后,就能将原数最右侧的1去掉

这种类型的题目:

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

题解:

 public int hammingWeight(int n) {int count = 0;while(n != 0){n &= n-1;count++;}return count;}}​

(2)比特位计数. - 力扣(LeetCode)

题解:

 class Solution {public int[] countBits(int n) {int[] ret = new int[n+1];for(int i = 0; i <= n; i++){int tmp = i;while(tmp != 0){tmp &= (tmp-1);ret[i]++;}}return ret;}}

(3)汉明距离:. - 力扣(LeetCode)

题解:

class Solution {public int hammingDistance(int x, int y) {x = x ^ y;int count = 0;while(x != 0){x &= (x-1);count++;}return count;}}
1.8异或运算的运算律

实际上我们用前面讲到的不进位相加就能理解

(1)a ^ 0 = a

(2)a ^ a = 0(形象称为消消乐)

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

那么不进位相加实际上就是每一列抵消1的过程,既然这样,那就和顺序毫无关系了,因此满足交换律

这种类型的题目:

(1)只出现一次的个数:. - 力扣(LeetCode)

题解:

 class Solution {public int singleNumber(int[] nums) {int ret = 0;for(int x : nums){ret ^= x;}return ret;}}

(2)只出现一次的数字 III:

题目:. - 力扣(LeetCode)

解析:如果将给定数组里面的所有数都异或起来,那么最后得到的数一定是一个非0的数,因为如果为0那么这两个只出现一次数就相等了.我们将这两个数字记作m,n,数组全部异或后得到x.

如果我们提取出x的二进制表示中最低位的1,假设是第k位.那么根据异或操作等效于不进位相加,m n的第k位一定只有一个是1,另一个第k位一定是0;而在数组中,除了这两个数以外,其他数都是成对出现的,数组里面的任何数也可以分成两派,一派第k位是1,另一派第k位是0;如果我们能把数组里面的元素按照这两派分开,那么m,n一定是在不同的派,由于其他数都是成对出现的,那么分别将两派里面的元素全部异或,就能得到m和n

那么接下来的重点就在于如何分派,我们前面已经提取出x的最低位的1,这里设为tmp,可以利用tmp进行分派,如果一个数的第k位是0,那么这个数 & tmp结果就是0;反之则是1,这样就能分派了

最后,还有一个细节问题,我们前面说过提取最低位的1方法是 n & (n-1),但是如果n刚好是int类型的最小值,即二进制:10000000,00000000,00000000,00000000,因为这个数进行-n操作后还是他本身,所以我们就不能提取出最后一个n,而恰恰这个数提取出最低位的1后就是他本身,那么就不用提取了

题解:

 class Solution {public int[] singleNumber(int[] nums) {int x = 0;for(int t : nums){x ^= t;}int tmp = (x == Integer.MIN_VALUE ? x : x & (-x));int type1 = 0;int type2 = 0;for(int t : nums){if((t & tmp)== 0){type1 ^= t;}else{type2 ^= t;}}return new int[]{type1,type2};}}

2.丢弃的数字

题目:. - 力扣(LeetCode)

2.1解析

此题就是在一段连续区间内找到缺失的那一个数.假设给定的数组长度是x,由于数组是缺少了一个数,那么完全的区间应该是0<= x <= n.

对于此类问题我们利用哈希表当然是做得出来,但是这类题利用位运算的方法更加巧妙

我们前面说过,两个相同的数进行异或操作得到的是0

我们前也做过一道题:只出现一次的数字,即在数组中,除了一个数只出现了一次,其余数字都出现了两次,我们将这个数组元素全部异或起来就能得到这个只出现一次的数

在这道题:如果我们将原数组 和 完全的区间结合起来形成一个新的数组,那问题不就转化为只出现一次的数字的问题了吗,因为在新数组中,除了确失的那个数以外,其余数都出现了两次.

2.2题解
 class Solution {public int missingNumber(int[] nums) {int ret = 0;int n = nums.length;for(int i = 0; i < n+1; i++){ret ^= i;}for(int x : nums){ret ^= x;}return ret;}}

3.两整数之和

题目:. - 力扣(LeetCode)

3.1解析

在做这道题之前,有一个重要的前提知识(我在做这道题之前实际上是不知道的):

两个整数相加,实际上对应的二进制相加的结果就是两个整数相加的结果

既然题目要求不能用+和-,那么我们就利用位运算来计算,那么二进制的加法实际上也分成两个步骤:相加 和 进位

(1)对应位置相加

我们前面说过异或运算实际上就是不进位相加,这不就是正好满足我们的要求吗

那么我们就可以先把要相加的两个数进行异或,得到的就是未进位的和

(2)进位操作

实际上进位操作无非就是两个1相加那就进1,那么我们的&运算不正好满足对应二进制位两个为1才为1吗,只不过&运算的结果是在原位置,但是无妨,我们将得到的结果 << 1位即可

最后将(1)的得到的数 与 (2)得到的数 再次进行^操作,此时还可能会有进位,因此这个操作我们要循环进行,直到(2)操作得到的数字为0,说明相加结果没有进位了,那么循环结束

那么这样的话,我们就用二进制的位运算来代替了原来的加减法

3.2题解
 class Solution {public int getSum(int a, int b) {int m = a ^ b;int n = (a & b) << 1;while(n != 0){int tmp = m ^ n;n = (m & n) << 1;m = tmp;}return m;}}

4.只出现一次的数字II

题目:. - 力扣(LeetCode)

4.1解析

假设题目给我们的是如图所示的数组,只有一个元素出现了1次,其他的都出现了3次.我们将数组元素十进制转化为二进制,就会发现,如果我们将所有元素的某一个二进制位加起来,得到的结果去%3,得到的就一定是[只出现一次]的那个数的对应比特位.比如:我们将所有元素的第0比特位相加,那么就会得到4,4 % 3 = 1,那么99的第0位比特位就是0

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

5.消失的两个数字

题目:. - 力扣(LeetCode)

5.1解析

实际上这道题实际上就是(丢失的数字)和(只出现一次的数字 III)

如果给定数组的长度是n,那么完全的数据范围就是 1 <= x <= n+2.在(只出现一次的数字 III)中,数组元素中,有两个出现了一次,其余都出现了两次;那么在这道题中,我们就可以利用(丢失的数字)的思想来将这道题转化为(只出现一次的数字 III)的题目,即把我们完全的数字范围与 给定的数组合成一个新数组.假设我们的数组是:[2,3],那么数据范围就是1,2,3,4,那么新的数组就是:1,2,2,3,3,4.在这个数组中利用(只出现一次的数字 III)的思想来解决这道题

5.2题解
class Solution {public int[] missingTwo(int[] nums) {int k = 0;int n = nums.length;for(int i = 1; i <= n+2; i++){k ^= i;}for(int x : nums){k ^= x;}k = (k == Integer.MIN_VALUE ? k : k & (-k));int type1 = 0,type2 = 0;for(int i = 1; i <= n+2; i++){if((k & i) == 0){type1 ^= i;}else{type2 ^= i;}}for(int x : nums){if((k & x) == 0){type1 ^= x;}else{type2 ^= x;} }return new int[]{type1,type2};}}

感谢您的访问!!

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

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

相关文章

vue 部署 abap BSP

How to Create a Vue.Js App with VS Code and Deploy... - SAP Community 详情见上面这个教程连接 Open VS Code and open a terminal window and run npm install -g vue/cli restart vscode. Open a terminal window again and go to your workspace folder to create the…

国密算法 SM9 公钥加密 数字签名 密钥交换 基于身份的密码算法(IBC)完整高效的开源python代码

上篇文章&#xff08;发布于2023-09-18&#xff09;给自己挖了个坑&#xff0c;说是要搞定SM9。从国庆前一周开始&#xff0c;到现在一个月时间&#xff0c;这个坑终于填上了。此前信息安全数学基础太差&#xff0c;理解不了SM9双线性对、扩域计算等等&#xff0c;为此还特意选…

由浅到深认识C语言(6):变量的存储类型

该文章Github地址&#xff1a;https://github.com/AntonyCheng/c-notes 在此介绍一下作者开源的SpringBoot项目初始化模板&#xff08;Github仓库地址&#xff1a;https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址&#xff1a;https://blog.csdn…

想做好国际现货黄金行情交易 请从这样的预期开始

国际现货黄金行情以多变而著称&#xff0c;而这样多变的行情&#xff0c;又由于其趋势稳定&#xff0c;而深受投资者喜爱。我们看近期&#xff0c;国际现货黄金行情不断上涨&#xff0c;短线呈现出很强的上涨趋势&#xff0c;这正是投资者入场交易的好时机&#xff0c;但是要做…

目标检测——PP-YOLOE-R算法解读

PP-YOLO系列&#xff0c;均是基于百度自研PaddlePaddle深度学习框架发布的算法&#xff0c;2020年基于YOLOv3改进发布PP-YOLO&#xff0c;2021年发布PP-YOLOv2和移动端检测算法PP-PicoDet&#xff0c;2022年发布PP-YOLOE和PP-YOLOE-R。由于均是一个系列&#xff0c;所以放一起解…

从头手搓一台ros2复合机器人(带机械臂)

一.前言 大家好呀&#xff0c;从本小节开始我们就步入了仿真篇&#xff0c;主要对机器人仿真进行介绍与操作&#xff0c;当然仿真有优点也有缺陷&#xff0c;基于对此学习&#xff0c;我们可以对上几小节创建的小车模型模拟硬件的特性&#xff0c; 比如&#xff1a; 有多重…

洛谷[NOIP2016 提高组] 玩具谜题

[NOIP2016 提高组] 玩具谜题 题目背景 NOIP2016 提高组 D1T1 题目描述 小南有一套可爱的玩具小人&#xff0c;它们各有不同的职业。 有一天&#xff0c;这些玩具小人把小南的眼镜藏了起来。小南发现玩具小人们围成了一个圈&#xff0c;它们有的面朝圈内&#xff0c;有的面…

JetBrains学生包续期

Pycharm进入显示your license has expired 这个意思是你的许可过期了&#xff0c;重新输入最新的激活码就可以了。 1. 说明我的JetBrains学生包需要进行续期了。首先登录JetBrains官网打开个人主页&#xff0c;点击Renew my Education Pack 2. 然后填写一系列信息&#xff0c;…

VR历史建筑漫游介绍|虚拟现实体验店|VR设备购买

VR历史建筑漫游是一种利用虚拟现实技术&#xff0c;让用户可以身临其境地参观和探索历史建筑的体验。通过VR头显和相关设备&#xff0c;用户可以在虚拟环境中自由移动和互动&#xff0c;感受历史建筑的真实氛围和文化内涵。 在VR历史建筑漫游中&#xff0c;您可以选择不同的历史…

手机备忘录怎么导出到电脑,如何将手机备忘录导出到电脑

备忘录是我们日常生活和工作中常用的工具之一&#xff0c;我们可以在手机上轻松地记录重要的事务、想法和灵感。然而&#xff0c;在某些情况下&#xff0c;我们可能需要将手机备忘录导出到电脑进行更详细的整理和管理。那么&#xff0c;手机备忘录怎么导出到电脑&#xff0c;如…

开箱即用之 windows部署jdk、设置nginx、jar自启

jdk安装 官网下载对应的安装包&#xff0c;解压之后放在本地指定的文件夹下 传送门https://www.oracle.com/java/technologies/downloads/#jdk21-windows 我比较喜欢下载zip方式的&#xff0c;解压之后直接能用&#xff0c;不需要安装了 配置环境 JAVA_HOME 添加path路径 …

MySQL数据导入的方式介绍

MySQL数据库中的数据导入是一个常见操作&#xff0c;它涉及将数据从外部源转移到MySQL数据库表中。在本教程中&#xff0c;我们将探讨几种常见的数据导入方式&#xff0c;包括它们的特点、使用场景以及简单的示例。 1. 命令行导入 使用MySQL命令行工具mysql是导入数据的…

有关【指针运算】的经典笔试题

题目1&#xff1a; #include <stdio.h>int main() {int a[5] { 1, 2, 3, 4, 5 };int* ptr (int*)(&a 1);printf("%d,%d", *(a 1), *(ptr - 1));return 0; } //程序的输出结果是什么&#xff1f; 2 5解析&#xff1a; 代码的内存布局如图&#xff1a…

Nutanix 国产化替代|一文了解 SmartX 超融合替代可行性与迁移方案

2022 年 8 月 19 日&#xff0c;Nutanix&#xff08;路坦力&#xff09;宣布中国市场自 2023 财年起将转型为合作伙伴销售主导模式&#xff0c;引起了广泛关注&#xff1b;同时结合当前 IT 基础架构的国产化趋势背景&#xff0c;不少正在使用和考虑使用 Nutanix 产品的企业开始…

编译原理-实现LR语法分析器——沐雨先生

实验任务&#xff1a; 实现LR语法分析器 实验要求&#xff1a; 根据编译原理理论课教材中例3.22给出的算术表达式文法以及该文法的LR分析表&#xff0c;用C语言编写接受算术表达式为输入的语法分析器&#xff0c;以控制台&#xff08;或文本文件&#xff0c;也可以结合词法分…

C语言种sizeof()和strlen的区别

sizeof 是 C 语言内置的操作符关键字&#xff0c;而 strlen 是 C 语言库函数&#xff1b; sizeof 仅用于计算数据类型的大小或者变量的大小&#xff0c;而 strlen 只能以结尾为 \0 的字符串作为参数&#xff1b; 编译器在编译时就计算出了 sizeof 的结果&#xff0c;而 strlen …

Spring Security的开发

文章目录 1,介绍2, 核心流程3, 核心原理3.1 过滤器链机制3.2 主体3.3 认证3.4 授权3.5 流程图4, 核心对象4.1 UserDetailsService 接口4.2 PasswordEncoder 接口4.3 hasAuthority方法4.4 hasAnyAuthority方法4.5 hasRole方法4.5 hasAnyRole方法5, 核心注解5.1 @PreAuthorize5.1…

【Linux】系统开启和关闭过程

Linux 系统启动过程 BIOS 自检&#xff1a;在计算机开机时&#xff0c;BIOS 会进行自检&#xff0c;检查硬件设备是否正常。 加载引导程序&#xff1a;BIOS 自检完成后&#xff0c;会加载引导程序&#xff0c;如 GRUB、LILO 等。引导程序会加载内核和初始化 RAM 磁盘&#xff…

想入门Web测试,看这篇文章!

今天要谈的是很多软件测试工程师都需要面对的——Web测试 不管你是处在二十不惑的青春有你阶段还是三十而已的乘风破浪阶段我们都需要面对“Web测试”。 Web测试其实有以下几个方面&#xff1a; 1、页面测试 大多数的Web网站的网页都是html语言编写的&#xff0c;测试工程师…

3.3 RK3399项目开发实录-板载Ubuntu系统的使用(wulianjishu666)

嵌入式物联网常用90款传感器开发例程。链接&#xff1a;https://pan.baidu.com/s/1oisHMZXDzKqa4EspY83V-A?pwdo5f4 1. 介绍 Ubuntu 使用手册是针对 Firefly 官方发布的 Ubuntu 系统固件特性所编写&#xff0c;适用于 Ubuntu Desktop 与 Minimal 系统&#xff0c;部分与 UI 显…