【C语言】位操作符:一个数组中只有两个数字是出现一次,其他所有数字都出现了两次,找出这两个数字。

一、题目:

一个数组中只有两个数字是出现一次,其他所有数字都出现了两次,找出这两个数字。


二、解题思路一:遍历法

这个方法是将数组遍历两遍,找出只出现一次的数,并输出。
这个方法的缺点是时间复杂度高,为O(N^2),程序并不高效。

源码:
#include<stdio.h>
#include<Windows.h>
#include<assert.h>

void find_data(int arr[], int size){
    int i = 0, j = 0;
    int flag = 0;
    printf("data is : ");
    for (; i < size; i++){
        flag = 0;
        for (j = 0; j < size; j++){
            if (arr[j] == arr[i] && i != j)
                flag = 1;
        }
        if (0 == flag)
            printf(" %d  ", arr[i]);

    }
    printf("\n");
}


int main(){
    int arr[] = { 1, 2, 4, 3, 1, 5, 2, 3};
    int len = 0;
    len = sizeof(arr) / sizeof(arr[0]);
    find_data(arr, len);
    system("pause");
    return 0;
}
程序运行结果:

这里写图片描述


三、解题思路二:异或法

位操作基本概念

按位与运算符(&):

仅当两个操作数都为1时,结果为1,否则为0。参与运算的数以补码方式出现。

【特殊用法】:

  • 清零。如果想将一个单元清零,即使其全部二进制位为0,只要与一个各位都为零的数值相与,结果为零。
  • 取一个数中指定位。找一个数,对应X要取的位,该数的对应位为1,其余位为零,此数与X进行“与运算”可以得到X中的指定位。

按位或运算符(|):

仅当两个操作数都为0时,结果为0,否则为1。

【特殊用法】:

  • 常用来对一个数据的某些位置1。找到一个数,对应X要置1的位,该数的对应位为1,其余位为零。此数与X相或可使X中的某些位置1。

异或运算符(^):

参加运算的两个对象,如果两个相应位为“异”(值不同),则该位结果为1,否则为0。

【特殊用法】:

  • 使特定位翻转 。找一个数,对应X要翻转的各位,该数的对应位为1,其余位为零,此数与X对应位异或即可。
  • 与0相异或,保留原值。

取反运算符(~):

对一个二进制数按位取反,即将0变1,1变0。

左移运算符(<<):

将一个运算对象的各二进制位全部左移若干位(左边的二进制位丢弃,右边补0)。

若左移时舍弃的高位不包含1,则每左移一位,相当于该数乘以2。

右移运算符(>>):

将一个数的各二进制位全部右移若干位,正数左补0,负数左补1,右边丢弃。

操作数每右移一位,相当于该数除以2。

解题思路
程序源码
#include<stdio.h>
#include<Windows.h>
#include<assert.h>

#if 1
void find_data(int arr[], int size){
    assert(arr);
    assert(size > 0);

    int res = arr[0];
    int i = 1;
    int flag = 1;
    int x = 0, y = 0;

    //tep 1
    for (; i < size; i++){
        res ^= arr[i];
    }

    //tep 2
    for (i = 0; i < 32; i++){
        if (res & (flag <<= i))
            break;
    }

    //tep 3
    for (i = 0; i < size; i++){
        if (flag & arr[i])
            x ^= arr[i];
        else
            y ^= arr[i];
    }

    printf("%d , %d \n", x, y);
}
#endif


int main(){
    int arr[] = { 1, 2, 4, 3, 1, 5, 2, 3};
    int len = 0;
    len = sizeof(arr) / sizeof(arr[0]);
    find_data(arr, len);
    system("pause");
    return 0;
}
运行结果:

这里写图片描述

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值