Find a duplicate element in an array

Question:

Suppose you have an array of 1001 integers. The integers are in random order, but you know each of the integers is between 1 and 1000 (inclusive). In addition, each number appears only once in the array, except for one number, which occurs twice. Assume that you can access each element of the array only once. Describe an algorithm to find the repeated number. If you used auxiliary storage in your algorithm, can you find an algorithm that does not require it?

 

如果可以用辅助空间的话,可以用HashSet来处理,如果add失败说明有重复的。但是如果不能用额外空间的话呢?

 

Solution 1:

考虑加法。1+2+...+n = n(n-1)/2

假设数组所有元素之和为sum, 那么sum-n(n-1)/2就是我们要找的重复的数。

public int findDuplicate(int[] nums) {
	int sum = 0;
	int n = nums.length;
	int expected = n*(n-1)/2;
	for(int i=0; i<n; i++) {
		sum += nums[i];
	}
	return sum-expected;
}

但是数组的元素数达到或接近INT_MAX的时候, 加法运算就会溢出。我们还可以考虑位运算XOR。

XOR具有以下性质:

交换律A \oplus B = B \oplus A

结合律A \oplus (B \oplus C)=(A \oplus B) \oplus C

恒等律X\oplus 0=X

归零律X\oplus X=0

自反A \oplus B \oplus B=A\oplus 0=A

假设给定数组A[6] = {5,2,3,4,3,1},而我们期望的数组应该为A[6]={0,1,2,3,4,5}。其中我们用0来代替重复的元素。

接下来对给定数组和我们期望数组的所有元素做异或运算:

(5^0)^(2^1)^(3^2)^(4^3)^(3^4)^(1^5)

= (5^5)^(2^2)^(3^3)^(4^4)^(1^1)^(3^0)

= 0^3

= 3

写成代码则非常简单。

Solution2:

public int findDuplicate(int[] nums) {
	int dup = 0;
	for(int i=0; i<nums.length; i++) {
		dup ^= nums[i]^i;
	}
	return dup;
}

Reference:

http://stackoverflow.com/questions/2605766/how-to-find-a-duplicate-element-in-an-array-of-shuffled-consecutive-integers 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值