287. Find the Duplicate Number
Your runtime complexity should be less than O(n2).
There is only one duplicate number in the array, but it could be repeated more than once.
问题分析
1、暴力枚举
2、对数组排序,寻找相同的元素,时间复杂度O(nlgn)+O(n)
3、二分法,利用抽屉原理,简单表述为如果 在0~n/2中的数大于n/2,那么重复的数在0~n/2之间,依次类推,时间复杂度O(nlgn)
4、类似链表中的环,
假设数组中没有重复,那我们可以做到这么一点,就是将数组的下标和1到n每一个数一对一的映射起来。比如数组是213,则映射关系为0->2, 1->1, 2->3。假设这个一对一映射关系是一个函数f(n),其中n是下标,f(n)是映射到的数。如果我们从下标为0出发,根据这个函数计算出一个值,以这个值为新的下标,再用这个函数计算,以此类推,直到下标超界。
但如果有重复的话,这中间就会产生多对一的映射,比如数组2131,则映射关系为0->2, {1,3}->1, 2->3。
所以该题实际上就是找环路起点的题,和Linked List Cycle II一样。我们先用快慢两个下标都从0开始,快下标每轮映射两次,慢下标每轮映射一次,直到两个下标再次相同。这时候保持慢下标位置不变,再用一个新的下标从0开始,这两个下标都继续每轮映射一次,当这两个下标相遇时,就是环的起点,也就是重复的数
You must not modify the array (assume the array is read only).
You must use only constant, O(1) extra space.Your runtime complexity should be less than O(n2).
There is only one duplicate number in the array, but it could be repeated more than once.
问题分析
1、暴力枚举
2、对数组排序,寻找相同的元素,时间复杂度O(nlgn)+O(n)
3、二分法,利用抽屉原理,简单表述为如果 在0~n/2中的数大于n/2,那么重复的数在0~n/2之间,依次类推,时间复杂度O(nlgn)
4、类似链表中的环,
假设数组中没有重复,那我们可以做到这么一点,就是将数组的下标和1到n每一个数一对一的映射起来。比如数组是213,则映射关系为0->2, 1->1, 2->3。假设这个一对一映射关系是一个函数f(n),其中n是下标,f(n)是映射到的数。如果我们从下标为0出发,根据这个函数计算出一个值,以这个值为新的下标,再用这个函数计算,以此类推,直到下标超界。
但如果有重复的话,这中间就会产生多对一的映射,比如数组2131,则映射关系为0->2, {1,3}->1, 2->3。
所以该题实际上就是找环路起点的题,和Linked List Cycle II一样。我们先用快慢两个下标都从0开始,快下标每轮映射两次,慢下标每轮映射一次,直到两个下标再次相同。这时候保持慢下标位置不变,再用一个新的下标从0开始,这两个下标都继续每轮映射一次,当这两个下标相遇时,就是环的起点,也就是重复的数
class Solution {
public:
int findDuplicate(vector<int>& nums) {
int slow = 0;
int fast = 0;
// 找到快慢指针相遇的地方
do{
slow = nums[slow];
fast = nums[nums[fast]];
} while(slow != fast);
int find = 0;
// 用一个新指针从头开始,直到和慢指针相遇
while(find != slow){
slow = nums[slow];
find = nums[find];
}
return find;
}
};