异或法巧解数组中两独数

题目分析

该问题要求在数组中找到两个只出现一次的数字,其余数字均成对出现。需要在O(n)时间复杂度和O(1)空间复杂度下完成。

核心思路:异或法

异或运算具有以下关键性质:

  • a ^ a = 0
  • a ^ 0 = a
  • 异或满足交换律和结合律

实现步骤

将所有数组元素进行异或操作,最终结果为两个只出现一次的数字的异或值(成对的数字异或后抵消为0)。

找到异或结果中最低位的1,该位置是两数字二进制表示中不同的位。

根据该位将数组分为两组,分别进行异或操作,最终得到两个只出现一次的数字。

代码实现

c 复制代码
#include <stdio.h>

void FindNum(int arr[], int len, int *pnum1, int *pnum2) {
    int ret = 0;
    for (int i = 0; i < len; i++) {
        ret ^= arr[i];
    }

    int pos = 0;
    for (int i = 0; i < 32; i++) {
        if ((ret >> i) & 1) {
            pos = i;
            break;
        }
    }

    *pnum1 = 0;
    *pnum2 = 0;
    for (int i = 0; i < len; i++) {
        if ((arr[i] >> pos) & 1) {
            *pnum1 ^= arr[i];
        } else {
            *pnum2 ^= arr[i];
        }
    }
}

int main() {
    int arr[] = {1, 2, 3, 4, 5, 1, 2, 3, 4, 6};
    int len = sizeof(arr) / sizeof(arr[0]);
    int num1 = 0;
    int num2 = 0;
    FindNum(arr, len, &num1, &num2);
    printf("%d %d\n", num1, num2);
    return 0;
}

运行结果

程序输出为:

复制代码
5 6

两个数字的顺序可能互换,不影响结果正确性。

关键知识点

  • sizeof(arr)/sizeof(arr[0]):计算数组长度,仅在数组定义所在函数内有效。
  • 按位异或^:成对的数字异或后抵消为0,最终得到两个单次出现数字的异或值。
  • 分组异或:利用两个数字二进制表示中的不同位,将数组分成两组分别异或。
相关推荐
JieE21210 小时前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack2018 小时前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树20 小时前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
JieE2121 天前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2121 天前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术2 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦2 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
用户497863050732 天前
(一)小红的数组操作
算法·编程语言
怕浪猫2 天前
Electron 系列文章封面图
算法·架构·前端框架