程序员面试宝典(34)-找出数组中两个只出现一次的数字

原创 2015年05月04日 18:23:16

题目:

一个整型数组里除了两个数字之外,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。要求时间复杂度是O(n),空间复杂度是O(1)。

分析:

首先我们考虑这个问题的一个简单版本:一个数组里除了一个数字之外,其他的数字都出现了两次。请写程序找出这个只出现一次的数字。

这个题目的突破口在哪里?题目为什么要强调有一个数字出现一次,其他的出现两次?我们想到了异或运算的性质:任何一个数字异或它自己都等于0。也就是说,如果我们从头到尾依次异或数组中的每一个数字,那么最终的结果刚好是那个只出现依次的数字,因为那些出现两次的数字全部在异或中抵消掉了。

有了上面简单问题的解决方案之后,我们回到原始的问题。如果能够把原数组分为两个子数组。在每个子数组中,包含一个只出现一次的数字,而其他数字都出现两次。如果能够这样拆分原数组,按照前面的办法就是分别求出这两个只出现一次的数字了。

我们还是从头到尾依次异或数组中的每一个数字,那么最终得到的结果就是两个只出现一次的数字的异或结果。因为其他数字都出现了两次,在异或中全部抵消掉了。由于这两个数字肯定不一样,那么这个异或结果肯定不为0,也就是说在这个结果数字的二进制表示中至少就有一位为1。我们在结果数字中找到第一个为1的位的位置,记为第N位。现在我们以第N位是不是1为标准把原数组中的数字分成两个子数组,第一个子数组中每个数字的第N位都为1,而第二个子数组的每个数字的第N位都为0。

现在我们已经把原数组分成了两个子数组,每个子数组都包含一个只出现一次的数字,而其他数字都出现了两次。因此到此为止,所有的问题我们都已经解决。

核心代码

public void findNumsAppearOnce(int[] data, int length) {
        if (length < 2)
            return;
        int resultExclusiveOR = 0;
        for (int i = 0; i < length; i++)
            resultExclusiveOR ^= data[i];   //两个只出现一次的数字异或后的值
        //得到resultExclusiveOR中第一个为1的位的位置
        int indexOf = findFirstBitIs1(resultExclusiveOR);
        int num1 = 0;
        int num2 = 0;
        for (int i = 0; i < length; i++) {
            //将数组分为两组,一组中数字的第indexOf位为1,另一组中数字的第indexOf位不为1
            if (IsBit1(data[i], indexOf) == 1)
                num1 ^= data[i];
            else
                num2 ^= data[i];
        }
        System.out.println(num1 + " " + num2);
    }

    /**
     * 在结果数字中找到第一个为1的位的位置
     * @param resultExclusiveOR 两个只出现一次的数字异或后的值
     * @return
     */
    public int findFirstBitIs1(int resultExclusiveOR) {
        int num = 1;
        while ((resultExclusiveOR & 1) != 1) {
            resultExclusiveOR = resultExclusiveOR >> 1;
            num++;
        }
        return num;
    }

    /**
     * 判断该数字的第indexOf位是否为1
     * @param i 数字i
     * @param indexOf 结果数字中的第一个为1的位的位置
     * @return
     */
    public int IsBit1(int i, int indexOf) {
        i = i >> indexOf;
        return i & 1;
    }

剑指Offer面试题40(Java版):数组出现一次的数字

题目:一个整型数组里除了两个数字之外,其他的数字都出现了两次。  * 请些程序找出这两个只出现一次的数字。要求时间复杂度为O(n),空间复杂度为O(1) 例如输入数组{2,4,3,6,3,2,5,...
  • jsqfengbao
  • jsqfengbao
  • 2015年08月10日 19:04
  • 1724

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

一个数组中只有两个数字是出现一次,其他所有数字都出现了两次。找出这两个数字,编程实现。 本题思路: 1.将0与数组每个元素依次亦或的结果就是两个只出现一次的元素亦或的值,相同元素的二进制位是相同的,...
  • qq_23556715
  • qq_23556715
  • 2016年08月19日 14:53
  • 1898

剑指offer:数组中只出现一次的数字(java)

题目:一个整型数组里除了两个数字之外,其他的数字都出现了两次。请些程序找出这两个只出现一次的数字。要求时间复杂度为O(n),空间复杂度为O(1).     如输入数组{2,4,3,6,3,2,5,...
  • abc7845129630
  • abc7845129630
  • 2016年10月06日 19:03
  • 1138

整型数组里只出现一次的数字--[c语言笔试题]

找出数组内只出现一次的数字,是一系列的笔试题,来考察大家对位运算的掌握,下面我们从最简单的开始来看看吧! 一、题目:一个整型数组里只有一个数字出现了一次,其余数字都出现了两次,请写程序找出出现了一次的...
  • XHfight
  • XHfight
  • 2016年06月19日 15:00
  • 1303

查找数组中只出现一次的一个数

对于一个数组其中每个数出现了两次,只有一个数出现了一次,找出这个只出现了一次的一个数,这是一个经典的面试题,把数组中的数进行异或运算,出现两次的数异或的结果为0,最终异或的结果就为只出现了一次的一个数...
  • xhp1564881
  • xhp1564881
  • 2016年04月06日 10:04
  • 1692

java 找出数组中只出现一次的数字

题目:一个整型数组里除了两个数字之外,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。 算法如下:import java.util.HashMap; import java.util.It...
  • diu_brother
  • diu_brother
  • 2016年04月01日 21:00
  • 1565

数组中只出现一次的数字 C++

【题目】 一个整型数组里除了两个数字之外,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。 【题目分析】 注意异或运算的特点:即两个相同的数异或结果为0。也就是说,当一个整型数组...
  • noc_lemontree
  • noc_lemontree
  • 2016年07月29日 16:42
  • 262

【算法编程】找出只出现一次的数-singleNumber

题目意思:一个数值数组中,大部分的数值出现两次,只有一个数值只出现过一次,求编程求出该数字。 要求,时间复杂度为线性,空间复杂度为O(1).解题思路:1.先排序,后查找。由于排序的最快时间是O(nl...
  • chenriwei2
  • chenriwei2
  • 2015年07月02日 22:07
  • 1162

一个数组中,除了有两个数字只出现一次外,其他数字都出现两次,求出这两个出现一次的数字

求出两个只出现一次的数字 首先,我们知道两个相同的数字进行异或操作时为0,如果题目里面只有一个数字出现一次的话,我们就可以直接对数组中的所有数字进行异或操作,最后得到的数字就是单独出现的那个数字,但是...
  • u013412497
  • u013412497
  • 2016年08月12日 11:17
  • 789

位运算之只出现一次的的数字

****************位运算系列之数组中只出现一次的数字****************** //题目(1):在一个数组中只有一个数字出现一次,其他数字都是成对出现的!让你找出这...
  • bitboss
  • bitboss
  • 2016年06月06日 11:15
  • 875
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:程序员面试宝典(34)-找出数组中两个只出现一次的数字
举报原因:
原因补充:

(最多只允许输入30个字)