在一个数组中只有两个数字出现一次,其余数字都是成对出现,编写函数找出这两个数字。
示例:输入:1 2 3 4 3 2 1 5
输出:4 5
方法一、暴力求解
设置一个标识符tmp为1,利用两个指针遍历判断整个数组,用continue跳过自身和自身的判断,如果数组后面有和自身相等的,则将标识符改为0,最后利用标识符作为判断条件打印单身狗数字。
void Find_signal(int* p, int sz)
{
int i = 0;
int* flag = p;
int j = 0;
for (j = 0; j < sz; j++)
{
int tmp = 1;
for (i = 0; i < sz; i++)
{
if (i == j)
{
continue;
}
if (*(p + j) == *(flag + i))
{
tmp = 0;
break;
}
}
if (tmp==1)
{
printf("%d\n", *(p + j));
}
}
}
总结:暴力求解方法的问题在于它并不是获得两个数字,而是直接打印出来,对于只想获取到数字的某些情况来说此方法并不适用。
方法二、异或
我们知道按(二进制)位异或操作符(^),相同为0,相异为1。所以两个相同的数字异或是0,且0与任何一个数字异或还是那个数字。
所以首先我们可以对数组中的所有元素进行异或,这样实际上就是数组中两个不同的元素进行异或,如1,2,3,6,1,2,3,5进行异或如下
接着将整个数组的元素分成两组,将两个单身狗数字分别放在两组之中。那么对数组该怎么分组呢,以什么条件进行分组呢。
以两个数字二进制的第n位不同作为区分,如上图5和6的二进制的倒数第二位不同,所以可以将数组中元素二进制的倒数第二位是0的放在一组中(1,1,3,3,5),是1的放在一组中(2,2,6)。
接着对两组数分别进行全部异或,这样就找到了两个数字。
void Find_signal(int arr[], int sz, int* dog1, int* dog2)
{
int i = 0;
int ret = 0;
//异或
for (i = 0; i < sz; i++)
{
ret = ret ^ arr[i];
}
//计算ret的二进制中左右边的第几位1
int pos = 0;
for (pos = 0; pos < 32; pos++)
{
if (((ret >> pos) & 1) == 1)
{
break;
}
}
//分组
for (i = 0; i < sz; i++)
{
if (((arr[i] >> pos) & 1) == 1)
{
*dog1 ^= arr[i];
}
else
{
*dog2 ^= arr[i];
}
}
}
总结: 这种方法比较好,可以将两个单身狗数字通过指针赋值过去,直接获得。