一次遍历找出不同的数(位运算)

原文链接

题目:

输入一组偶数个的数,其中只有一对数是不同的,其他的都是成对出现。
数字的范围: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;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值