给定一个包含 n + 1 个整数的数组 nums,其数字都在 1 到 n 之间(包括 1 和 n),可知至少存在一个重复的整数。假设只有一个重复的整数,找出这个重复的数。
示例 1:
输入: [1,3,4,2,2]
输出: 2
示例 2:
输入: [3,1,3,4,2] 输出: 3
说明:
- 不能更改原数组(假设数组是只读的)。
- 只能使用额外的 O(1) 的空间。
- 时间复杂度小于 O(n2) 。
- 数组中只有一个重复的数字,但它可能不止重复出现一次。
这道题比较好做哈哈哈,主要思路就是新建一个数组来储存每一个数出现的次数,这样可以做到线性时间内找到出现的重复数,题目要求只有一个重复数,所以减少了很多难度,下面给出AC代码:
class Solution {
public:
int findDuplicate(vector<int>& nums) {
int res;
int n=nums.size();
int a[n]={0};
for(int i=0;i<n;i++){
a[nums[i]]++;
}
for(int i=0;i<n;i++){
if(a[nums[i]]>=2){
res=nums[i];
break;
}
}
return res;
}
};
看了一下讨论,发现学到了很多东西,下面我把讨论的内容复制粘贴在这里供学习使用
【笔记】这道题(据说)花费了计算机科学界的传奇人物Don Knuth
24小时才解出来。并且我只见过一个人(注:Keith Amling
)用更短时间解出此题。
快慢指针,一个时间复杂度为O(N)
的算法。
-
其一,对于链表问题,使用快慢指针可以判断是否有环。
-
其二,本题可以使用数组配合下标,抽象成链表问题。但是难点是要定位环的入口位置。
举个例子:nums = [2,5, 9 ,6,9,3,8, 9 ,7,1]
,构造成链表就是:2->[9]->1->5->3->6->8->7->[9]
,也就是在[9]
处循环。
-
其三,快慢指针问题,会在环内的
[9]->1->5->3->6->8->7->[9]
任何一个节点追上,不一定是在[9]
处相碰,事实上会在7
处碰上。 -
其四,必须另起一个
for
循环定位环入口位置[9]
。这里需要数学证明。