跳到主要内容
极客日志极客日志面向AI+效率的开发者社区
首页博客GitHub 精选镜像工具UI配色美学隐私政策关于联系
搜索内容 / 工具 / 仓库 / 镜像...⌘K搜索
注册
博客列表
C++算法

算法位运算实战:两整数之和、只出现一次数字及消失数字

位运算解决三道经典算法题。第一题利用异或无进位加法与按位与进位循环求解两数之和。第二题统计比特位总和模三还原唯一数字。第三题通过异或分组将缺失两数问题转化为单数查找。代码均使用 C++ 实现,时间复杂度优化至线性。

晚风告白发布于 2026/3/27更新于 2026/6/422 浏览
算法位运算实战:两整数之和、只出现一次数字及消失数字

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;
    }
};
算法总结及流程解析

文章配图

37.消失的两个数字

题目链接

面试题 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};
    }
};
算法总结及流程解析

文章配图

目录

  1. 35.两个整数之和
  2. 题目链接
  3. 题目描述
  4. 题目示例
  5. 解法(位运算)
  6. 算法思路
  7. C++算法代码
  8. 算法总结及流程解析
  9. 36.只出现一次的数字 ||
  10. 题目链接
  11. 题目描述
  12. 题目示例
  13. 解法(比特位计数)
  14. 算法思路
  15. C++算法代码
  16. 算法总结及流程解析
  17. 37.消失的两个数字
  18. 题目链接
  19. 题目描述
  20. 题目示例
  21. 解法(位运算)
  22. 算法思路
  23. C++算法代码
  24. 算法总结及流程解析
  • 💰 8折买阿里云服务器限时8折了解详情
  • Magick API 一键接入全球大模型注册送1000万token查看
  • 🤖 一键搭建Deepseek满血版了解详情
  • 一键打造专属AI 智能体了解详情
极客日志微信公众号二维码

微信扫一扫,关注极客日志

微信公众号「极客日志V2」,在微信中扫描左侧二维码关注。展示文案:极客日志V2 zeeklog

更多推荐文章

查看全部
  • OpenClaw 接入 QVeris:让 AI 助手具备实时数据查询能力
  • Windows 系统安装 Neo4j 图数据库图文教程
  • 使用 Trae 插件 Builder 模式开发端午包粽子小游戏
  • C++ 哈希表核心机制:冲突处理与负载因子
  • 5 款主流开源 AI Agent 框架深度解析
  • GPT-OSS-20B 模型本地部署及 WebUI 交互指南
  • UniApp 打包鸿蒙应用流程与系统权限配置
  • AI 大模型的十大发展趋势预判
  • C++ 类和对象(中)
  • Flutter 集成 google_generative_language_api 适配鸿蒙实现 AI
  • ChatGPT 如何利用结构化原则实现高效信息管理
  • 前端核心知识点梳理与面试复习
  • 基于视觉的增强现实特效技术解析
  • AI“代笔”的困境与破局:百考通AI如何理性应对论文查重与AIGC检测
  • 前端面试高频场景题汇总
  • C++ STL 源码解析:基于红黑树实现 map 和 set
  • 2025 年世界职业院校技能大赛人工智能赛道备赛方案
  • 医疗 NLP 实战:从电子病历分析到智能问答
  • 空洞卷积(Dilated Convolution)网络架构与 PAMAP2 数据集实验分析
  • 国内 8 个利用 AI 技能变现的在线兼职渠道

相关免费在线工具

  • 加密/解密文本

    使用加密算法(如AES、TripleDES、Rabbit或RC4)加密和解密文本明文。 在线工具,加密/解密文本在线工具,online

  • Gemini 图片去水印

    基于开源反向 Alpha 混合算法去除 Gemini/Nano Banana 图片水印,支持批量处理与下载。 在线工具,Gemini 图片去水印在线工具,online

  • Base64 字符串编码/解码

    将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online

  • Base64 文件转换器

    将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online

  • Markdown转HTML

    将 Markdown(GFM)转为 HTML 片段,浏览器内 marked 解析;与 HTML转Markdown 互为补充。 在线工具,Markdown转HTML在线工具,online

  • HTML转Markdown

    将 HTML 片段转为 GitHub Flavored Markdown,支持标题、列表、链接、代码块与表格等;浏览器内处理,可链接预填。 在线工具,HTML转Markdown在线工具,online