使用位操作高效解决单个元素出现问题【位运算】

2024-09-03 19:44

本文主要是介绍使用位操作高效解决单个元素出现问题【位运算】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

使用位操作高效解决单个元素出现问题

在日常的算法面试和编程挑战中,常常会遇到寻找单个出现元素的问题。尽管可以用哈希表(map)轻松解决,但要求更高效的线性时间复杂度和常量空间复杂度时,位操作特别是异或(XOR)运算提供了一个巧妙的解决方案。本篇博客将深入探讨这个问题,详细解释异或运算的特性,并展示其在解决该类问题中的强大作用。

问题描述:

给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次。找到那个只出现了一次的元素。

示例:

  • 输入: nums = [4,1,2,1,2]
  • 输出: 4

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

常规解法:使用 map 记录次数

最初,我们可能会想到使用一个哈希表来记录每个元素的出现次数,然后遍历哈希表找到那个只出现一次的元素。具体步骤如下:

  1. 创建一个哈希表,遍历数组并记录每个元素的出现次数。
  2. 再次遍历哈希表,找到只出现一次的元素并返回。

这种方法虽然直观且易于理解,但由于需要额外的存储空间来保存元素的出现次数,空间复杂度为 (O(n))。

代码实现:

#include <iostream>
#include <vector>
#include <unordered_map>using namespace std;class Solution {
public:int singleNumber(vector<int>& nums) {unordered_map<int, int> countMap;for (int num : nums) {countMap[num]++;}for (auto& entry : countMap) {if (entry.second == 1) {return entry.first;}}return -1;  // Ideally, this line should never be reached.}
};int main() {Solution sol;vector<int> nums = {4, 1, 2, 1, 2};cout << "The single number is: " << sol.singleNumber(nums) << endl;return 0;
}

分析:

  • 时间复杂度:由于我们遍历了数组两次,因此时间复杂度为 (O(n))。
  • 空间复杂度:需要 (O(n)) 的额外空间来存储哈希表。

尽管这种方法解决了问题,但它不符合题目要求的常量空间复杂度。为此,我们需要寻找更高效的解决方案。

高效解法:使用异或运算

异或运算的性质:

  • 交换律a ^ b = b ^ a
  • 结合律a ^ (b ^ c) = (a ^ b) ^ c
  • 任何数与0异或等于它本身a ^ 0 = a
  • 任何数与自己异或等于0a ^ a = 0

基于这些性质,我们可以推导出一个重要结论:如果对数组中所有的元素进行异或运算,那么出现两次的元素都会相互抵消,最终的结果就是那个只出现了一次的元素

具体步骤:
  1. 初始化一个变量 res 为0。
  2. 遍历整个数组,将每个元素与 res 进行异或运算。
  3. 遍历结束后,res 中存储的就是那个只出现了一次的元素。
代码实现:
#include <iostream>
#include <vector>using namespace std;class Solution {
public:int singleNumber(vector<int>& nums) {int res = 0;for (int num : nums) {res ^= num;  // 进行异或运算}return res;}
};int main() {Solution sol;vector<int> nums = {4, 1, 2, 1, 2};cout << "The single number is: " << sol.singleNumber(nums) << endl;return 0;
}

详细解释:

  • nums = [4, 1, 2, 1, 2],初始 res = 0
  • 遍历数组并依次异或每个元素:
    • res = 0 ^ 4 = 4
    0000
^ 0100
_______0100
  • res = 4 ^ 1 = 5
   0100
^ 0001
_______0101
  • res = 5 ^ 2 = 7
   0101
^ 0010
_______0111
  • res = 7 ^ 1 = 6
   0111
^ 0001
_______0110
  • res = 6 ^ 2 = 4
   0110
^ 0010
_______0100
  • 最终结果 res = 4,即那个只出现了一次的元素。
性能分析:

时间复杂度:
由于我们只遍历了一次数组,所以时间复杂度为 (O(n)),与使用哈希表的方法相同。

空间复杂度:
我们只使用了一个额外的变量 res,所以空间复杂度为 (O(1)),这比哈希表的方法更优。

异或运算在其他场景中的应用:

异或运算不仅在寻找单个出现元素的问题中有应用,还可以用于以下场景:

1. 交换两个变量的值

在C++或其他支持按位运算的语言中,可以使用异或运算来在没有额外存储空间的情况下交换两个变量的值。这是因为对于任何数字 ab,有以下性质:

  • a ^ b ^ b = a (两次异或同一个数等于原数)
  • a ^ a = 0 (任何数与自己异或等于0)

基于这两个性质,可以编写如下代码来交换两个变量 ab

void swapWithoutTemp(int &a, int &b) {a = a ^ b;  // a现在存储的是a^bb = a ^ b;  // b现在存储的是a (因为a^b^b=a)a = a ^ b;  // a现在存储的是b (因为a^b^a=b)
}

2. 检测两个数的不同位

异或运算可以帮助我们找到两个整数之间不同的二进制位。如果两个位相同,则异或结果为0;如果不同,则结果为1。因此,通过异或运算,我们可以很容易地找出两个数的二进制表示中哪些位不同。

bool bitsDiffer(int a, int b) {int diff = a ^ b;while (diff != 0) {if (diff & 1) {// 当前位不同std::cout << "Bit differs at position: " << 31 - __builtin_clz(diff) << std::endl;}diff >>= 1;}
}

这里使用了__builtin_clz函数来找到最高位1的位置,并计算出具体哪一位不同。__builtin_clz返回的是从左边开始第一个非零位的位置,所以需要减去这个值以得到具体的位位置。

3. 求两个数的汉明距离

汉明距离是指两个字符串或数字的二进制表示中对应位不同的数量。使用异或运算,然后计算结果中1的个数,就可以得到汉明距离。

int hammingDistance(int x, int y) {int xorResult = x ^ y;int distance = 0;while (xorResult) {distance += xorResult & 1; // 如果最低位为1,则增加距离xorResult >>= 1; // 移除最低位}return distance;
}

这段代码首先计算出xy之间的异或结果,然后通过逐位检查并计数结果中的1来得出汉明距离。这种方法适用于任何整数类型。

总结:

在处理数组中找出唯一的单个出现元素的问题时,异或运算提供了一个高效且优雅的解决方案。相比于使用哈希表记录次数的方法,异或运算不仅能保证线性时间复杂度,还能将空间复杂度降至常量。

通过理解和掌握异或运算的性质,我们不仅能更好地解决这一类问题,还能将其应用到其他相关的算法场景中,大大提升算法编写的效率和性能。

这篇关于使用位操作高效解决单个元素出现问题【位运算】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1133865

相关文章

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

Makefile简明使用教程

文章目录 规则makefile文件的基本语法:加在命令前的特殊符号:.PHONY伪目标: Makefilev1 直观写法v2 加上中间过程v3 伪目标v4 变量 make 选项-f-n-C Make 是一种流行的构建工具,常用于将源代码转换成可执行文件或者其他形式的输出文件(如库文件、文档等)。Make 可以自动化地执行编译、链接等一系列操作。 规则 makefile文件

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

高效+灵活,万博智云全球发布AWS无代理跨云容灾方案!

摘要 近日,万博智云推出了基于AWS的无代理跨云容灾解决方案,并与拉丁美洲,中东,亚洲的合作伙伴面向全球开展了联合发布。这一方案以AWS应用环境为基础,将HyperBDR平台的高效、灵活和成本效益优势与无代理功能相结合,为全球企业带来实现了更便捷、经济的数据保护。 一、全球联合发布 9月2日,万博智云CEO Michael Wong在线上平台发布AWS无代理跨云容灾解决方案的阐述视频,介绍了

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi