[剑指Offer]40.数组中只出现一次的数字

题目

一个整型数组里除了两个数字之外,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。

思路

我们直到异或的性质:

任何一个数字异或他自己都等于0.

所以说我们如果从头到尾依次异或每一个数字,那么最终的结果刚好只出现一次的数字,因为成对出现的两次的数字全部在异或中抵消了。

这道题中有两个数字只出现一次。这样的话我们得到的结果就是这两个数字的异或结果。因此我们想办法把原数组分成两个子数组,使得每个子数组包含一个只出现一次的数字。这样我们就可以对这两个子数组分别异或,就能得到两个只出现一次的数字。

现在问题是怎样把原数组分成两个子数组,使得每个子数组包含一个只出现一次的数字?

由于异或的结果肯定是那两个只出现一次的数字的异或结果,所以异或结果肯定不为0,也就是说这个结果二进制表示中至少有一位是1。我们可以找到第一个为1的下标位置(记为第n位)作为划分数组的依据。第一个子数组中每个数的第n位为1,第二个子数组的每个数的第n位为0。对于出现两次的数肯定分在同一个子数组中。那两个只出现一次的数字分别分在不同的子数组中。

假设数组为{2,4,3,6,3,2,5,5},异或结果为2,二进制表示为0010。异或得到的结果中倒数第二位为1,于是我们根据倒数第二位是不是为1分为两个子数组。第一个子数组{2,3,6,3,2},其中所有数字的倒数第二位为1;第二个子数组{4,5,5},其中所有数字的倒数第二位为0。接下来只要分别对这两个子数组求异或,就能找出第一个子数组中只出现一次的数字为6,第二个子数组中只出现一次的数字为4。

代码

/*---------------------------------------
*   日期:2015-04-26
*   作者:SJF0115
*   题目: 40.数组中只出现一次的数字
*   网址:http://www.nowcoder.com/books/coding-interviews/e02fdb54d7524710a7d664d082bb7811
*   结果:AC
*   来源:剑指Offer
*   博客:
-----------------------------------------*/
#include <iostream>
#include <vector>
using namespace std;

class Solution {
public:
    void FindNumsAppearOnce(vector<int> data,int* num1,int *num2) {
        int size = data.size();
        if(size <= 1){
            return;
        }//if
        int result = data[0];
        // 所有数XOR
        for(int i = 1;i < size;++i){
            result ^= data[i];
        }//for
        // 二进制中第一个为1的下标
        int index = 0;
        while((result != 0) && ((result & 1) == 0)){
            result >>= 1;
            ++index;
        }//while
        result = (1 << index);
        *num1 = 0;
        *num2 = 0;
        for(int i = 0;i < size;++i){
            // 第index位为1
            if((data[i] & result) > 0){
                *num1 ^= data[i];
            }//
            else{
                *num2 ^= data[i];
            }//else
        }//for
    }
};

int main(){
    vector<int> num = {3,1,10,1,3,6,2,6};
    int* num1 = new int(0);
    int* num2 = new int(0);
    Solution s;
    s.FindNumsAppearOnce(num,num1,num2);
    cout<<*num1<<" "<<*num2<<endl;
    return 0;
}
时间: 2024-08-04 16:21:17

[剑指Offer]40.数组中只出现一次的数字的相关文章

【剑指offer】数组中只出现一次的数字(1)

转载请注明出处:http://blog.csdn.net/ns_code/article/details/27649027 题目描述: 一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字. 输入: 每个测试案例包括两行: 第一行包含一个整数n,表示数组大小.2<=n <= 10^6. 第二行包含n个整数,表示数组元素,元素均为int. 输出: 对应每个测试案例,输出数组中只出现一次的两个数.输出的数字从小到大的顺序. 样例输入: 8 2 4 3 6 3

【剑指offer】数组中只出现一次的数字

一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字. /*考虑过程: 首先我们考虑这个问题的一个简单版本:一个数组里除了一个数字之外,其他的数字都出现了两次.请写程序找出这个只出现一次的数字. 这个题目的突破口在哪里?题目为什么要强调有一个数字出现一次,其他的出现两次?我们想到了异或运算的性质:任何一个数字异或它自己都等于0 .也就是说,如果我们从头到尾依次异或数组中的每一个数字,那么最终的结果刚好是那个只出现一次的数字,因为那些出现两次的数字全部在异或中

【剑指offer】数组中仅仅出现一次的数字(1)

转载请注明出处:http://blog.csdn.net/ns_code/article/details/27649027 题目描写叙述: 一个整型数组里除了两个数字之外,其它的数字都出现了两次.请敲代码找出这两个仅仅出现一次的数字. 输入: 每一个測试案例包括两行: 第一行包括一个整数n,表示数组大小.2<=n <= 10^6. 第二行包括n个整数,表示数组元素,元素均为int. 输出: 相应每一个測试案例.输出数组中仅仅出现一次的两个数.输出的数字从小到大的顺序. 例子输入: 8 2 4

【剑指offer】数组中只出现一次的数

题目描述 一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字. 分析: 经典的异或技巧题 两个相同的数字异或的结果为0,一个数和0异或的结果是其本身,假设现在那两个不同的数字是A和B,那么将整个数组的元素依次异或得到的结果ans就是A和B的异或结果 ans的二进制中第k位的1代表A的第k位和B的第k位不同,我们现在按照第k位是否相同将原数组分成两个子数组,那么必定A和B分别分散在两个子数组中,然后将子数组依次异或,得到的结果就是A和B的值! 至于k的取值则

剑指offer (36) 数组中的逆序对

题目:在数组中的两个数字如果前面一个数字大于后面一个数字,则这两个数字组成一个逆序对 题解分析: 首先应该想到很简单的一种解法,顺序遍历数组,对每个数,逐个比较该数字和其以后的数字,T(n) = O(n^2) (1)总体的意思就是将数组分成两段,首先求段内的逆序对数量,比如下面两段代码就是求左右两端数组段内的逆序对数量 count += Merge(data, temp, first, mid);//找左半段的逆序对数目 count += Merge(data, temp, mid + 1, e

【剑指offer】数组中的逆序对

# @left part: [start, mid] # @right part: (mid, end] def merge(data, start, mid, end): if mid < start or end < mid: return 0 reverse = 0 ''' @ for start, it play as the start index of left part, and mid @ play as the end index of left part; @ mid +

剑指OFFER之数组中出现次数超过一半的数字(九度OJ1370)

题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. 输入: 每个测试案例包括2行: 第一行输入一个整数n(1<=n<=100000),表示数组中元素的个数. 第二行输入n个整数,表示数组中的每个元素,这n个整数的范围是[1,1000000000]. 输出: 对应每个测试案例,输出出现的次数超过数组长度的一半的数,如果没有输出-1. 样例输入:

剑指offer (29) 数组中出现次数超过一半或1/3或1/N的数字

题目:数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字 方法一:如果把这个数字排序,那么排序之后位于数组中间的数字一定就是出现次数超过数组长度一半的数字 这个数字就是统计学中的中位数,即长度为n的数组中第n/2大的数字 在数组中得到任意第k大数字,这一问题有O(n)解,注:这里第kth个元素,kth从1开始计数,并且重复元素不去重 (1) 直接sort排序,然后定位到索引为kth-1的元素 int FindKth1(std::vector<int>& num, int kt

【Java】 剑指offer(39) 数组中出现次数超过一半的数字

本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1, 2, 3, 2, 2, 2, 5, 4, 2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. 思路 思路一:数字次数超过一半,则说明:排序之后数组中间的数字一定就是所求的数字. 利用partition()函数获得某一随机数字,其余数字按大小排在该数字的左右.若该