《算法题讲解指南:优选算法-位运算》--35.两个整数之和,36.只出现一次的数字 ||,37.消失的两个数字

《算法题讲解指南:优选算法-位运算》--35.两个整数之和,36.只出现一次的数字 ||,37.消失的两个数字

🔥小叶-duck个人主页

❄️个人专栏《Data-Structure-Learning》

《C++入门到进阶&自我学习过程记录》《算法题讲解指南》--从优选到贪心

未择之路,不须回头
已择之路,纵是荆棘遍野,亦作花海遨游


目录

35.两个整数之和

题目链接:

题目描述:

题目示例:

解法(位运算):

算法思路:

C++算法代码:

算法总结及流程解析:

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

题目链接:

题目描述:

题目示例:

解法(比特位计数):

算法思路:

C++算法代码:

算法总结及流程解析:

38. 消失的两个数字

题目链接:

题目描述:

题目示例:

解法(位运算):

算法思路:

C++算法代码:

算法总结及流程解析:

结束语


35.两个整数之和

题目链接:

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

题目描述:

题目示例:

解法(位运算):

算法思路:
  • 异或 ^ 运算本质是【无进位加法
  • 按位与 & 操作能够得到【进位】的对应位置,但还需要左移1才是需要进位的位置
  • 然后一直循环,直到【进位】变成 0 为止

C++算法代码:

class Solution { public: int getSum(int a, int b) { //解法:位运算(^异或:无进位相加) while(b) { int x = a ^ b; int y = (a & b) << 1; a = x; b = y;//得到进位的对应位置,再左移1才是需要进位的位置 //只进行一次a & b不一定保证新的a和b没有需要进位的位置,所以需要将这个步骤进行循环 //a & b为0则说明没有进位的位置了 } return a ^ b; } };

算法总结及流程解析:

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

题目链接:

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

题目描述:

题目示例:

解法(比特位计数):

算法思路:

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

C++算法代码:

class Solution { public: int singleNumber(vector<int>& nums) { //方法一:排序(时间复杂度较大:O(NlogN),但容易想到) // sort(nums.begin(), nums.end()); // for(int i = 0; i < nums.size() - 1; i += 3) // { // if(nums[i] != nums[i + 1]) // { // return nums[i]; // } // } // return nums.back(); //方法二:位运算(为线性时间复杂度:O(32N),但非常巧妙比较难想) int ret = 0;//作为位图 for(int i = 0; i < 32; i++) //一个数二进制位的长度,依次修改ret的每一位 { int sum = 0; for(int x = 0; x < nums.size(); x++) //计算nums中所有数的二进制第i位的和 { sum += ((nums[x] >> i) & 1); } ret |= ((sum % 3) << i); //求和结果余3后修改到ret对应第i位的位置 } return ret; } };

算法总结及流程解析:

38. 消失的两个数字

题目链接:

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

题目描述:

题目示例:

解法(位运算):

算法思路:

      本题相当于就是 268.丢失的数字 + 260.只出现一次的数字||| 组合起来的题。
      先将数组中的数【1,n+2】区间内的所有数异或】在一起,问题就变成了:有两个数出现了【一次】,其余所有的数出现了【两次】。进而变成了 260.只出现了一次的数字||| 这道题。

C++算法代码:

class Solution { public: vector<int> missingTwo(vector<int>& nums) { //位运算 //先将数组nums所有数以及1~N+2区间所有数全部异或一遍,得到消失两数的异或 int ret = 0; int a = 0; int b = 0; //分别表示两个消失的数 for(int i = 0; i < nums.size(); i++) { ret ^= nums[i]; } for(int i = 1; i <= nums.size() + 2; i++) { ret ^= i; } //此时ret = a ^ b //由于a和b一定不同,所以一定会存在某一位比特位一个是0一个是1 ////两者异或后对应的那一位就一定是1,所以我们需要找到那一位 //获取到ret最右侧出现1的比特位位置 int x = 0; while(1) { if(((ret >> x) & 1) == 1) { break; } x++; } //将数组nums所有数以及(1~N+2)区间所有数分成两类: //一类就是第x位比特位值为0,一类就是第x位比特位值为1 ////这样两个消失数就会被分开,通过异或,在数组以及(1~N+2)区间都出现的数就会抵消, //a和b也就分别是这两个消失数了 for(int i = 0; i < nums.size(); i++) { if(((nums[i] >> x) & 1) == 0)//假设a是第x位比特位为0的消失数 { a ^= nums[i]; } else //假设b就是第x位比特位为1的消失数 { b ^= nums[i]; } } for(int i = 1; i <= nums.size() + 2; i++) { if(((i >> x) & 1) == 0) { a ^= i; } else { b ^= i; } }//两次循环则数组中存在的数异或了两次被抵消,a和b就分别是两个消失数 return {a, b}; } };

算法总结及流程解析:

结束语

      到此,35.两个整数之和,36.只出现一次的数字 ||,37.消失的两个数字 这三道算法题就讲解完了。第35题通过异或和按位与操作实现无进位加法,循环处理进位直至为零,高效求解两数之和;第36题利用比特位计数技术,统计所有数字各二进制位出现次数,模3结果定位唯一出现一次的数字;第37题通过位运算巧妙解决。首先将数组与[1,n+2]区间所有数异或,转化为两个数出现一次的问题。核心思路借鉴了260题《只出现一次的数字III》,通过提取不同比特位将数分组异或,最终找到缺失的两个数。希望大家能有所收获!

Read more

【Linux】Shell 脚本中的条件判断语句

【Linux】Shell 脚本中的条件判断语句

👋 大家好,欢迎来到我的技术博客! 📚 在这里,我会分享学习笔记、实战经验与技术思考,力求用简单的方式讲清楚复杂的问题。 🎯 本文将围绕Linux这个话题展开,希望能为你带来一些启发或实用的参考。 🌱 无论你是刚入门的新手,还是正在进阶的开发者,希望你都能有所收获! 文章目录 * Shell 脚本中的条件判断语句 🐚 * 一、什么是 Shell 条件判断?🎯 * 二、基础 if 语句结构 🧱 * 2.1 单分支 if * 2.2 双分支 if-else * 2.3 多分支 if-elif-else * 三、Java 对比示例 ☕ * 3.1 单分支 Java 示例 * 3.2 双分支 Java 示例 * 3.3 多分支

By Ne0inhk
一文读懂 Linux 互斥锁:小白也能看懂的临界区保护指南,手把手教你彻底告别多线程数据“打架”

一文读懂 Linux 互斥锁:小白也能看懂的临界区保护指南,手把手教你彻底告别多线程数据“打架”

🔥海棠蚀omo:个人主页                 ❄️个人专栏:《初识数据结构》,《C++:从入门到实践》,《Linux:从零基础到实践》                 ✨追光的人,终会光芒万丈 博主简介: 目录 一.线程间互斥相关背景概念 二.互斥量mutex 2.1为什么会出现问题? 2.1.1判断条件 2.1.2ticket-- 2.2如何解决问题 2.2.1解决方式 2.2.2所产生的疑问 三.互斥实现原理探究 四.由线程互斥的缺点引出线程同步 前言: 在前面的章节中,我们了解了什么是线程,以及如何通过pthread库所提供的函数来对线程进行操作,但是我们要了解的不止有这些,我们创建多线程是为了让它们协作帮助我们去完成任务。 而今天及后面的章节我们就将目光聚焦在线程之间协作的方面,了解线程之间协作时会出现什么样的问题以及如何解决,下面就开始我们今天的内容。 一.线程间互斥相关背景概念 共享资源临界资源:多线程执⾏流被保护的共享的资源就叫做临界资源临界区:

By Ne0inhk
技能提升必备:鸿蒙HarmonyOS应用开发者认证

技能提升必备:鸿蒙HarmonyOS应用开发者认证

技能提升必备:鸿蒙HarmonyOS应用开发者认证,HarmonyOS 认证是华为为开发者打造的能力衡量体系。随着 HarmonyOS 系统影响力不断扩大,市场对相关开发人才需求激增。该认证分为基础与高级等不同级别,覆盖应用开发、设备开发等方向。通过认证,开发者能系统掌握 HarmonyOS 知识与技能,提升个人职业竞争力,为鸿蒙生态繁荣贡献力量,在万物智联时代获得更多发展机遇 。 技能提升必备:鸿蒙HarmonyOS应用开发者认证 🔆 在新时代的软件开发中,HarmonyOS 应用开发技术占据重要地位。随着 HarmonyOS 系统的广泛应用,招聘市场对这类开发者的需求越来越多。鸿蒙 HarmonyOS 应用开发者认证分为基础认证和高级认证两个级别,目的是帮助开发者系统掌握 HarmonyOS 的开发框架、API 调用、界面设计等基本技能,同时深入理解分布式技术原理,掌握跨设备协同、场景化服务等高级功能。 🔆 官方打造了针对不同角色、技术领域和业务场景的认证,让开发者能证明自己的专业水平和能力。其中,和 HarmonyOS 应用开发相关的认证有基础认证和高级认证,还有一些

By Ne0inhk
ARM Linux 驱动开发篇--- Linux 并发与竞争全解析(原子操作/自旋锁/信号量/互斥体)--- Ubuntu20.04

ARM Linux 驱动开发篇--- Linux 并发与竞争全解析(原子操作/自旋锁/信号量/互斥体)--- Ubuntu20.04

🎬 渡水无言:个人主页渡水无言 ❄专栏传送门: 《linux专栏》《嵌入式linux驱动开发》《linux系统移植专栏》 ❄专栏传送门: 《freertos专栏》《STM32 HAL库专栏》 ⭐️流水不争先,争的是滔滔不绝  📚博主简介:第二十届中国研究生电子设计竞赛全国二等奖 |国家奖学金 | 省级三好学生 | 省级优秀毕业生获得者 | ZEEKLOG新星杯TOP18 | 半导纵横专栏博主 | 211在读研究生 在这里主要分享自己学习的linux嵌入式领域知识;有分享错误或者不足的地方欢迎大佬指导,也欢迎各位大佬互相三连 目录 前言 一、并发与竞争核心概念 1.1、什么是并发与竞争? 1.2 Linux并发产生的4大原因(记牢!面试常问) 1.3 临界区与保护核心(重点!) 二、原子操作 2.1 原子操作简介 2.2 原子整形操作API 2.3 原子位操作API

By Ne0inhk