在一个数组中仅出现一次,其他数均出现两次,这个出现一次的数就被称为“单身狗“。
一.一个单身狗
我们知道异或运算操作符 ^ ,它的特点是对应二进制位相同为 0
,相异为 1
。
由此我们容易知道两个相同的数,进行异或运算得到的结果一定为 0,0和非0数字异或的结果为非0数字,因此我们可以将数组中的所有元素都进行异或,出现过两次的数异或结果将为0,留下来的就是单身狗了。
代码实现:
int FindSingle(int* arr,int sz)
{
int dog = 0;
int i = 0;
for (i = 0; i < sz; i++)
{
dog ^= arr[i];
}
return dog;
}
int main()
{
int arr[5] = { 1,4,2,1,2 };
int sz = sizeof(arr) / sizeof(arr[0]);
printf("单身狗为:%d\n", FindSingle(arr, sz));
return 0;
}
二.两个单身狗
如果数列中存在两个单身狗,依然和上面一样全部进行异或运算显然是得不到答案的,相同的数通过异或消除了,得到的会是两个单身狗异或的结果。
能不能将两个单身狗分开,在两个数组中分别以上面的方式找出单身狗呢?
异或的条件是对应二进制位相同为 0
,相异为 1
。通过两个单身狗数异或的结果,我们可以得到两个单身狗数在某些二进制位上单身狗的值不同(0或1),可以通过这位上的值不同来将两个单身狗分开。
同样,对于出现过两次的非单身狗数,也可以通过判断某一二进制位相同,将其放入同一数组中,再对该数组进行异或运算后消除。
代码实现:
void FindSingle(int* arr, int sz,int* dog,int* dog1,int* dog2)
{
int i = 0;
for (i = 0; i < sz; i++)
{
//全部异或得到两个单身狗的异或结果
*dog ^= arr[i];
}
//两个单身狗数某二进制位上的值不同
int pos = 0;
for (i = 0; i < 4; i++)
{
//dog的值为两个单身狗数异或的结果,dog的某一二进制位为1则代表两个单身狗在这一二进制位上不相等
//找出这一位置并拷贝下来
if (((*dog >> i) & 1) == 1)
{
pos = i;
break;
}
}
//将数组按pos位上的值为1或0分组并求异或
for (i = 0; i < sz; i++)
{
if (((arr[i] >> pos) & 1) == 1)
{
*dog1 ^= arr[i];
}
else
{
*dog2 ^= arr[i];
}
}
}
int main()
{
int arr[10] = { 1,2,3,4,5,1,2,3,4,6 };
int dog = 0;
int dog1 = 0;
int dog2 = 0;
int sz = sizeof(arr) / sizeof(arr[0]);
FindSingle(arr, sz, &dog, &dog1, &dog2);
printf("单身狗1是:%d,单身狗2是:%d", dog1, dog2);
return 0;
}