目录
题目:
一个数组中只有两个数字是出现一次,其他所有数字都出现了两次。
编写一个函数找出这两个只出现一次的数字
思路:
我们首先考虑暴力求解 使用遍历 但是遍历的复杂度应该是n^2
因为其他所有的数字都出现了两次 这时我们考虑或许可以用异或
针对这个题,我们可以先写出两个单身狗的异或和
然后我们就可以知道哪一位为1 再根据其中一位为1 对所有数组进行分类
两个单身狗肯定在两个不同的类别中 再分别求两个类的异或和 就是两个单身狗
具体实现:
写出基本框架:
void find_single_dog(int arr[],int sz)
{
}
int main()
{
int arr[]={1,2,3,4,5,1,2,3,4,6};
int sz= sizeof(arr)/sizeof(arr[0]);
find_single_dog();
return 0;
}
两个函数的异或和:
int i = 0;
int ret =0 ;//创建临时变量 承载异或和
for( i=0;i<sz;i++)
{
ret ^= arr[i];
}
求出异或和中哪一位为1
int pos = 0; //创建临时变量 承载那一位
for(i=0;i<32;i++)
{
if(((ret>>i)&1)==1)
{
pos = i;
break; //退出循环, 一位即可
}
}
根据具体那一位为1 分类 进而分别求出两个单身狗
for (i = 0; i < sz; i++)
{
if (((arr[i] >> pos) & 1) == 0)
{
single1 ^= arr[i];
}
}
single1 = ret ^single2;
进而输出就好了 。
最终代码:
void findTwoNum(int arr[], int n, int * pnum1, int * pnum2)
{
int i;
int sum = 0;
for (i = 0; i < 9; i++)
{
sum ^= arr[i];
} //先找到两个数互相异或的结果
int pos;
for (i = 0; i < 32; i++)
{
if (sum & 1 << i)
{
pos = i;
break;
}
} //再找到有分歧的一位。在这一位上,两个数一定是一个1一个0
*pnum1 = *pnum2 = 0;
for (i = 0; i < 10; i++)
{
if (arr[i] & 1 << pos)
{
*pnum1 ^= arr[i]; //这一位是1的,放在数1里
}
else
{
*pnum2 ^= arr[i]; //这一位是0的,放在数2里
}
}
}