【本文链接】

http://www.cnblogs.com/hellogiser/p/find-n-numbers-which-appear-only-once-in-array.html

题目】

一个数组中有三个数字a、b、c只出现一次,其他数字都出现了两次。请找出三个只出现一次的数字。

分析

这是一道很新颖的关于位运算的面试题。在之前的博文34.数组中2个只出现一次的数字[Find two numbers which appear once]中分析了N=1和N=2的情况。

(1).N=1时,数组所有数字异或的结果即为a。

(2).N=2时,数组所有数字异或的结果等于a^b,根据a^b的二进制中最后一个1出现的位置,将数组分为2组;对每一组数字进行异或,即可求得a和b。

(3).N=3时,数组所有数字异或的结果等于a^b^c。此时该如何区分呢?如果我们能够找出其中一个只出现一次的数字,剩下两个只出现一次的数字就可以转换为N=2的情况。

具体思路如下:

(1). f(x) = x & (-x)所得的结果即是x最后一位1所在的位置。

(2). x = a ^ b ^ c。

(3). flag = f(x^a)^f(x^b)^f(x^c) 结果必有一位是1,因为f(m)^f(n)结果为0或者为2个1。

(4). f(x^a)^f(x^b)^f(x^c)的第m位为1,则x^a, x^b, x^c必有1个或者3个第m位为1。

(5). 用反证法可得,x^a, x^b, x^c只有一个第m位为1。

举个例子data={1,2,3,4,4,5,5,6,6}

x = a ^ b ^ c =1^2^3 = 000

x^a=001, x^b=010, x^c=011

f(x^a)=001, f(x^b)=010, f(x^c)=001

flag = f(x^a)^f(x^b)^f(x^c)=010,flag = f(flag)=010

f(x^b)==flag

first=b=1

second=3,third=1

完整代码如下:

代码】

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
 
// 58_FindNumbersAppearOnce.cpp : Defines the entry point for the console application.
//
/*
    version: 1.0
    author: hellogiser
    blog: http://www.cnblogs.com/hellogiser
    date: 2014/5/27
*/

#include "stdafx.h"

// find number which appear once
void Find1NumberAppearOnce(int data[], int length, int &num)
{
    )
        return;

// get the exclusive or result of array
    // a
;
    ; i < length; ++i)
        xor ^= data[i];
    num = xor;
}

// get last 1 bit of n
// n=00001110--->00000010
unsigned int GetLast1Bit(int n)
{
    return n & (-n);
}

// find 2 numbers which appear once
void Find2NumbersAppearOnce(int data[], int length, int &num1, int &num2)
{
    )
        return;

// get the exclusive or result of array
    // a^b
;
    ; i < length; ++i)
        xor ^= data[i];

// find the last bit 1 of xor
    int flag = GetLast1Bit(xor);
    num1 = num2 = ;
    ; j < length; ++j)
    {
        // divide numbers in data into 2 groups by flag:
        // numbers in group1: the & result is 1
        // numbers in group2: the & result is 0
        if (flag & data[j])
        {
            num1 ^= data[j];
        }
        else
        {
            num2 ^= data[j];
        }
    }
}

// swap a and b
void myswap(int &a, int &b)
{
    int t = a;
    a = b;
    b = t;
}

// find 3 numbers which appear once
/*
(1). f(x) = x & (-x)
(2). x = a ^ b ^ c
(3). flag = f(x^a)^f(x^b)^f(x^c)
(4). flag = f(flag)
(5). x^a, x^b, x^c, only one of three is 1 at m-bit
  f(x^a)==flag

for example:
  data={1,2,3,4,4,5,5,6,6}
  x = a ^ b ^ c =1^2^3 = 000
  x^a=001, x^b=010, x^c=011
  f(x^a)=001, f(x^b)=010, f(x^c)=001
  flag = f(x^a)^f(x^b)^f(x^c)=010
  flag = f(flag)=010
  f(x^b)==flag
  first=b=1
  second=3,third=1
*/
void Find3NumbersAppearOnce(int data[], int length, int &num1, int &num2, int &num3)
{
    )
        return;

// get the exclusive or result of array
    // a^b^c
;
    ; i < length; i++)
        xor ^= data[i];

;
    ; i < length; i++)
        flag ^= GetLast1Bit(xor ^ data[i]);
    flag = GetLast1Bit(flag);

// get the first unique number
;
    ; i < length; i++)
        if(GetLast1Bit(data[i] ^ xor) == flag)
            first ^= data[i];
    num1 = first;

// move the first number to the end of array
; i < length; i++)
    {
        if (first == data[i])
        {
            myswap(data[i], data[length - ]);
            break;
        }
    }

// get the second and third unique number
, num2, num3);
}

//=================================================================
// test cases
void test_base1(int data[], int length)
{
    int num1;
    Find1NumberAppearOnce(data, length, num1);
    printf("%d\n", num1);
}

void test_base2(int data[], int length)
{
    int num1, num2;
    Find2NumbersAppearOnce(data, length, num1, num2);
    printf("%d %d\n", num1, num2);
}

void test_base3(int data[], int length)
{
    int num1, num2, num3;
    Find3NumbersAppearOnce(data, length, num1, num2, num3);
    printf("%d %d %d\n", num1, num2, num3);
}

void test_case1()
{
    };
    int length = sizeof(data) / sizeof(int);
    test_base1(data, length);
}

void test_case2()
{
    };
    int length = sizeof(data) / sizeof(int);
    test_base2(data, length);
}

void test_case3()
{
    };
    int length = sizeof(data) / sizeof(int);
    test_base3(data, length);
}

void test_main()
{
    test_case1(); // 1
    test_case2(); // 1 2
    test_case3(); // 2 3 1
}
//=================================================================

int _tmain(int argc, _TCHAR *argv[])
{
    test_main();
    ;
}

【参考】

http://www.cnblogs.com/hellogiser/p/3738909.html

http://zhedahht.blog.163.com/blog/static/25411174201283084246412/

http://www.cnblogs.com/youxin/p/3349834.html

http://www.cnblogs.com/kedebug/archive/2012/12/22/2829013.html

【本文链接】

http://www.cnblogs.com/hellogiser/p/find-n-numbers-which-appear-only-once-in-array.html

59. 总结篇:数组中N(n=1,2,3)个只出现一次的数字[find N numbers which appear only once in array]的更多相关文章

  1. 34.数组中2个只出现一次的数字[Find two numbers which appear once]

    [题目] 一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字.要求时间复杂度是O(n),空间复杂度是O(1). [分析] 这是一道很新颖的关于位运算的面试题. ...

  2. 转载——JavaScript学习笔记:取数组中最大值和最小值

    转载自:http://www.w3cplus.com/javascript/calculate-the-max-min-value-from-an-array.html. 取数组中最大值 可以先把思路 ...

  3. JavaScript学习:取数组中最大值和最小值

    在实际业务中有的时候要取出数组中的最大值或最小值.但在数组中并没有提供arr.max()和arr.min()这样的方法.那么是不是可以通过别的方式实现类似这样的方法呢?那么今天我们就来整理取出数组中最 ...

  4. leetcode-1 Two Sum 找到数组中两数字和为指定和

     问题描写叙述:在一个数组(无序)中高速找出两个数字,使得两个数字之和等于一个给定的值.如果数组中肯定存在至少一组满足要求. <剑指Offer>P214(有序数组) <编程之美& ...

  5. splice从数组中删除指定定数据

    /*从数组中删除指定定数据var somearray = ["mon", "tue", "wed", "thur"]so ...

  6. 剑指Offer 40. 数组中只出现一次的数字 (数组)

    题目描述 一个整型数组里除了两个数字之外,其他的数字都出现了偶数次.请写程序找出这两个只出现一次的数字. 题目地址 https://www.nowcoder.com/practice/e02fdb54 ...

  7. 剑指Offer - 九度1349 - 数字在排序数组中出现的次数

    剑指Offer - 九度1349 - 数字在排序数组中出现的次数2013-11-23 00:47 题目描述: 统计一个数字在排序数组中出现的次数. 输入: 每个测试案例包括两行: 第一行有1个整数n, ...

  8. 剑指offer笔记面试题4----二维数组中的查找

    题目:在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数. 测试用例: 二维数组中包含 ...

  9. 剑指offer第二版面试题1:数组中重复的数字(JAVA版)

    题目:在一个长度为n的数组里的所有数字都在0到n-1的范围内.数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复的次数.请找出数组中任意一个重复的数字.例如如果输入长度为7的数组{ ...

随机推荐

  1. [转载]ODAC (odp.net) 开发到部署

    1. 确定你开发机和服务器的操作系统是32位还是64位, 而且要确定要部署的服务器是什么操作系统; 2. 下载开发机和服务器所需的dll, 地址:http://download.csdn.net/de ...

  2. hdu3397 线段树 成段更新

    这题真的呵呵了.敲了很长时间,调了很多bug,把0 1 输出,解决了.最后想取反,怎么搞都有bug, 最后还是看了大牛们的博客.不过这题真的敲得爽,调bug时基本把线段树过程全部弄了一遍. #incl ...

  3. Java-日期转换

    如下: package 时间日期类; import java.text.SimpleDateFormat; import java.util.Date; public class 日期格式转换 { / ...

  4. jdownload的使用

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...

  5. 【UVA 1586】Ancient Cipher

    题 题意 给你一个只含CHON的有机物的化学式如C6H5OH求相对分子质量 分析 ... 代码 switch #include<cstdio> #include<cctype> ...

  6. Git删除文件操作

    使用Git删除文件需要使用Git rm命令来实现,最后git commit 需要注意的是直接rm命令删除后是不可以的,可以用git status 命令尝试一下,效果如图下(创建了test文件,演示了g ...

  7. ISO 基础之 (十三) protocol 协议

    一 简绍 protocol,简单来说就是一系列不属于任何类的方法列表,其中声明的方法可以被任何类实现.这种模式一般称为代理(delegation)模式.通过Protocol定义各种行为,在不同的场景采 ...

  8. H2嵌入式数据库

    一 H2 数据库 官网地址. http://www.h2database.com/html/cheatSheet.html

  9. Assembly文件被锁定

    使用 Assembly.LoadFile 加载程序集后 ,被加载的文件就会被锁定,之后就不能对其执行转移.删除等操作 为了解决次问题,我们可以先读取成字节流,然后转换成Assembly.代码如下:复制 ...

  10. Linux下如何搭建VPN服务器(转)

    VPN服务器的配置与应用 实验场景 通过将Linux配置VPN服务器允许远程计算机能够访问内网. 我的目的: 现在需要开发第三方接口,而第三方接口有服务器IP地址鉴权配置,这样在本地开发出来的程序每次 ...