题目:一个整型数组里除了两个数字之外。其它的数字都出现了两次,请敲代码找出这两个仅仅出现一次的数字。

要求时间复杂度是O(n),空间复杂度是O(1)。


举例说明

比如输入数组{2, 4, 3, 6, 3, 2, 5 },由于仅仅有4 、6 这两个数字仅仅出现一次,其它数字都出现了两次,所以输出4和6 。

解题思路

  这两个题目都在强调一个(或两个)数字仅仅出现一次,其它的出现两次。

这有什么意义呢?我们想到异或运算的一个性质:不论什么一个数字异或它自己都等于0。也就是说。 假设我们从头到尾依次异或数组中的每一个数字。那么终于的结果刚好是那个仅仅出现一次的数字,由于那些成对出现两次的数字所有在异或中抵消了。

  想明确怎么解决这个简单问题之后,我们再回到原始的问题,看看能不能运用同样的思路。

我们试着把原数组分成两个子数组,使得每一个子数组包括一个仅仅出现一次的数字。而其它数字都成对出现两次。假设可以这样拆分成两个数组, 我们就行依照前面的办法分别找出两个仅仅出现一次的数字了。

  我们还是从头到尾依次异或数组中的每一个数字,那么终于得到的结果就是两个仅仅出现一次的数字的异或结果。由于其它数字都出现了两次,在异或中所有抵消了。由于这两个数字肯定不一样,那么异或的结果肯定不为0。也就是说在这个结果数字的二进制表示中至少就有一位为1 。

我们在结果数字中找到第一个为1 的位的位置,记为第n 位。

如今我们以第n位是不是1为标准把原数组中的数字分成两个子数组,第一个子数组中每一个数字的第n 位都是1 , 而第二个子数组中每一个数字的第n 位都是0。

由于我们分组的标准是数字中的某一位是1 还是0 。 那么出现了两次的数字肯定被分配到同一个子数组。由于两个同样的数字的随意一位都是同样的,我们不可能把两个同样的数字分配到两个子数组中去,于是我们已经把原数组分成了两个子数组。每一个子数组都包括一个仅仅出现一次的数字,而其它数字都出现了两次。

我们已经知道怎样在数组中找出唯一一个仅仅出现一次数字。 因此到此为止所有的问题都已经攻克了。

代码实现

public class Test40 {
public static int[] findNumbersAppearanceOnce(int[] data) {
int[] result = {0, 0}; if (data == null || data.length < 2) {
return result;
} int xor = 0;
for (int i : data) {
xor ^= i;
} int indexOf1 = findFirstBit1(xor); for (int i : data) {
if (isBit1(i, indexOf1)) {
result[0] ^= i;
} else {
result[1] ^= i;
}
} return result;
} private static int findFirstBit1(int num) {
int index = 0;
while ((num & 1) == 0 && index < 32) {
num >>>= 1;
index++;
} return index;
} private static boolean isBit1(int num, int indexBit) {
num >>>= indexBit;
return (num & 1) == 1;
} public static void main(String[] args) {
int[] data1 = {2, 4, 3, 6, 3, 2, 5, 5};
int[] result1 = findNumbersAppearanceOnce(data1);
System.out.println(result1[0] + " " + result1[1]); int[] data2 = {4, 6};
int[] result2 = findNumbersAppearanceOnce(data2);
System.out.println(result2[0] + " " + result2[1]); int[] data3 = {4, 6, 1, 1, 1, 1};
int[] result3 = findNumbersAppearanceOnce(data3);
System.out.println(result3[0] + " " + result3[1]);
}
}

执行结果

【剑指Offer学习】【面试题40:数组中仅仅出现一次的数字】的更多相关文章

  1. 【剑指Offer】面试题03. 数组中重复的数字

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

  2. 《剑指offer》面试题03. 数组中重复的数字

    问题描述 找出数组中重复的数字. 在一个长度为 n 的数组 nums 里的所有数字都在 0-n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了几次.请找出数组中 ...

  3. 《剑指offer》面试题51. 数组中的逆序对

    问题描述 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数. 示例 1: 输入: [7,5,6,4] 输出: 5   限制: ...

  4. 《剑指offer》面试题39. 数组中出现次数超过一半的数字

    问题描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 你可以假设数组是非空的,并且给定的数组总是存在多数元素. 示例 1: 输入: [1, 2, 3, 2, 2, 2, 5, 4, ...

  5. [简单-剑指 Offer 53 - I. 在排序数组中查找数字 I]

    [简单-剑指 Offer 53 - I. 在排序数组中查找数字 I] 统计一个数字在排序数组中出现的次数. 示例 1: 输入: nums = [5,7,7,8,8,10], target = 8 输出 ...

  6. 剑指 Offer 53 - I. 在排序数组中查找数字 I + 二分法

    剑指 Offer 53 - I. 在排序数组中查找数字 I Offer_53_1 题目描述 方法一:使用HashMap package com.walegarrett.offer; /** * @Au ...

  7. 力扣 - 剑指 Offer 53 - I. 在排序数组中查找数字 I

    题目 剑指 Offer 53 - I. 在排序数组中查找数字 I 思路1 一般来说,首先想到的是使用一个变量,从头开始遍历整个数组,记录target数组出现的次数,但是这样的时间复杂度是O(n),还是 ...

  8. 剑指offer(一):二维数组中的查找

    说明: 1.本系列是根据<剑指Offer>这个系列做的一个小笔记. 2.直接动力是因为师兄师姐找工作很难,而且机械出生的我面试算法更难. 3.刚开始准备刷LeetCode.LintCode ...

  9. 剑指offer三十五之数组中的逆序对

    一.题目 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数P.并将P对1000000007取模的结果输出. 即输出P%1000 ...

  10. 剑指offer笔记面试题3----数组中重复的数字

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

随机推荐

  1. 粘滞位(sticky bit)

    linux特殊权限:setUid, setGid, 粘着位(sticky) (1)目录的X权限(执行) 文件的可执行权限很简单,也就是可否执行它的意思,但目录的执行权限又代表什么意思呢? 当然不可能是 ...

  2. ubuntu apt-get常用命令的使用

             packagename指代为软件包的名称 apt-get install packagename     安装一个新软件包(参见下文的aptitude) apt-get remove ...

  3. POJ 3264 Balanced Lineup 简单RMQ

    题目:http://poj.org/problem?id=3264 给定一段区间,求其中最大值与最小值的差. #include <stdio.h> #include <algorit ...

  4. POJ 1837 Balance 01背包

    题目: http://poj.org/problem?id=1837 感觉dp的题目都很难做,这道题如果不看题解不知道憋到毕业能不能做出来,转化成了01背包问题,很神奇.. #include < ...

  5. myeclipse启动项目时报:An internal error occurred during: "Launching TestSpring on Tomcat 7.x". java.lang.NullPointerException 解决方法

    如果出现了上述的错误按照如下的3个步骤解决: 1.首先关闭MyEclipse工作空间. 2.然后删除工作空间下的 “/.metadata/.plugins/org.eclipse.core.runti ...

  6. tomcat https 配置

    以前基本上笔者对于安全性考虑的并不多,最近因为saas平台要开始逐渐推广,所以需要开始逐渐加强xss/crsf/https等措施以避免潜在的安全性风险.本文简单的记录下tomcat下https的配置. ...

  7. 实现js的类似alert效果的函数

    这个简单的类似alert的函数,效果还行,至于css样式,那没的说了,笔者确实尽力了,如果读者觉得太烂,你可以随便改函数的样式的,反正,笔者觉得还可以,呵呵. <!DOCTYPE html PU ...

  8. [译]GotW #1: Variable Initialization

    原文地址:http://herbsutter.com/2013/05/09/gotw-1-solution/ 第一个问题强调的是要明白自己在写什么的重要性.下面有几行简单的代码--它们大多数之间都有区 ...

  9. 17.2.1 Replication Implementation Details 复制实现细节:

    17.2 Replication Implementation 复制是基于master server 跟踪所有改变到他的数据库(更新,删除等等)在它的binary log. binary log 作为 ...

  10. Learning WCF Chapter1 Summary

    SummaryThis chapter covered a lot of ground,beginning with a look at the purpose of WCF,the problems ...