数组中只出现一次的数字
来源:互联网 发布:开淘宝店收益怎么样 编辑:程序博客网 时间:2024/05/16 18:26
题目描述
一个整型数组里除了两个数字之外,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。
考虑过程:
首先我们考虑这个问题的一个简单版本:一个数组里除了一个数字之外,其他的数字都出现了两次。请写程序找出这个只出现一次的数字。
这个题目的突破口在哪里?题目为什么要强调有一个数字出现一次,其他的出现两次?我们想到了异或运算的性质:任何一个数字异或它自己都等于0 。也就是说,如果我们从头到尾依次异或数组中的每一个数字,那么最终的结果刚好是那个只出现一次的数字,因为那些出现两次的数字全部在异或中抵消掉了。
有了上面简单问题的解决方案之后,我们回到原始的问题。如果能够把原数组分为两个子数组。在每个子数组中,包含一个只出现一次的数字,而其它数字都出现两次。如果能够这样拆分原数组,按照前面的办法就是分别求出这两个只出现一次的数字了。
我们还是从头到尾依次异或数组中的每一个数字,那么最终得到的结果就是两个只出现一次的数字的异或结果。因为其它数字都出现了两次,在异或中全部抵消掉了。由于这两个数字肯定不一样,那么这个异或结果肯定不为0 ,也就是说在这个结果数字的二进制表示中至少就有一位为1 。我们在结果数字中找到第一个为1 的位的位置,记为第N 位。现在我们以第N 位是不是1 为标准把原数组中的数字分成两个子数组,第一个子数组中每个数字的第N 位都为1 ,而第二个子数组的每个数字的第N 位都为0 。
现在我们已经把原数组分成了两个子数组,每个子数组都包含一个只出现一次的数字,而其它数字都出现了两次。因此到此为止,所有的问题我们都已经解决
可以用位运算实现,如果将所有所有数字相异或,则最后的结果肯定是那两个只出现一次的数字异或
的结果,所以根据异或的结果1所在的最低位,把数字分成两半,每一半里都还有只出现一次的数据和成对出现的数据
这样继续对每一半相异或则可以分别求出两个只出现一次的数字
class Solution {
public:
void FindNumsAppearOnce(vector<int> data,int* num1,int *num2) {
if(data.size() < 2) return ;
int myxor = 0;
int flag = 1;
for(int i = 0 ; i < data.size(); ++ i )
myxor ^= data[i];
while((myxor & flag) == 0) flag <<= 1;
*num1 =0;//其实这个地方等于0就可以了,因为两两都会消掉肯定有剩下的就是那个了吧
*num2 = 0;
for(int i = 0; i < data.size(); ++ i ){
if((flag & data[i]) == 0) *num2 ^= data[i];
else *num1 ^= data[i];
}
}
};
0 0
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- 数组中只出现一次的数字
- SGU100 A+B
- hdu 5806 NanoApe Loves Sequence Ⅱ(乘法原理/尺取法)
- 装饰模式
- 在文件管理器剪切到一个文件到相同的路径下,直接变成了复制功能
- Git 基础学习教程
- 数组中只出现一次的数字
- VS Code1.4 搭建Golang的开发调试环境(遇到很多问题)
- java23种设计模式全解析
- PhoneWindowManager总结
- 《UNIX环境高级编程》四文件和目录读书笔记
- STM32之SPI_FLASH(实例)
- 257.[Leetcode]Binary Tree Paths
- Git之删除文件学习教程
- HDOJ 5806 NanoApe Loves Sequence Ⅱ