题目:
输入一组偶数个的数,其中只有一对数是不同的,其他的都是成对出现。
数字的范围:0~INT_MAX
数字的个数:2~2000000个
如输入1 2 3 2 2 3
输出1 2
使用数组记录出现次数,并且判断是否出现偶数次,效率较低。
使用异或规则:
a⊕b=b⊕a 交换律
a⊕0=a 与0异或不变
a⊕b⊕b=a 偶数次异或为结果0
a⊕(b⊕c)=(a⊕b)⊕c 结合律
如果只有一个数字单独出现运用③:
#include <stdio.h>
#include <stdlib.h>
int main()
{
int arr[] = { 1, 2, 3, 4, 1, 2, 3 };
int ret = 0;
int len = sizeof(arr) / sizeof(arr[0]);
for (int i = 0; i < len; i++)
{
ret ^= arr[i];//按位异或
}
printf("出现奇数次的数为:> %d\n", ret);
system("pause");
return 0;
}
如果是有两个数均出现了一次,那么异或的结果为这两个数的异或,所以现在就是从这个结果中去算出各自的值。根据异或的性质,如果该位为0,表示两位数在该位二进制相同;若该位为1,表示二进制位不同。
现在我们看如何将一组数分为每一组数都只有一个数出现过一次的两组数:首先我们对所有数进行异或,那么得到的就是两个出现奇次的那两个数的异或,比如{ 1, 2, 3, 4, 1, 2, 7, 3 },就得到7⊕4,这个数肯定不为0,我们找出这个数的二进制数的最右边的1(设最右边的1在第inter位),然后找出数组的每个元素的第inter位,并判断此位是1还是0,是1的为一组,是0的为一组,这样就分好了组,再利用上面的例子,就可得到出现奇数次的那两个数。
#include <stdio.h>
#include <stdlib.h>
int main()
{
int arr[] = { 1, 2, 3, 4, 1, 2, 7, 3 };
int len = sizeof(arr) / sizeof(arr[0]);
int ret = 0;
int inter = 0;
int retA = 0;
int retB = 0;
for (int i = 0; i < len; i++)
{
ret ^= arr[i];
}
/*找ret最右边的1*/
inter = ret - (ret&(ret - 1));
for (int i = 0; i < len; i++)
{
int a = (arr[i] >> (inter - 1)) % 2; //取出arr[i]的第inter位
if (a == 0)
{
retA ^= arr[i];
}
else
{
retB ^= arr[i];
}
}
printf("出现奇数次的两个数为:> %d,%d\n", retA, retB);
system("pause");
return 0;
}