287. Find the Duplicate Number

来源:互联网 发布:淘宝商城退出多少商家 编辑:程序博客网 时间:2024/05/16 04:43

题目

Given an array nums containing n + 1 integers where each integer is between 1 and n (inclusive), prove that at least one duplicate number must exist. Assume that there is only one duplicate number, find the duplicate one.

Note:

  • 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.

思路

本题给定一个大小为n+1的数组,数组里面的值的值域为[1n];那么请找出数组里面的重复项。这道题目如果仔细分析的话,本质上就是个快慢指针的问题。首先什么是快慢指针呢?

快慢指针

快慢指针其实是计算机系统原理里的知识点,快慢指针中的快慢指的是移动的步长,即每次向前移动速度的快慢。例如可以让快指针每次沿链表向前移动2,慢指针每次向前移动1次。
通过结合应用,我们发现快慢指针有两大总结:

  • 判断循环链表,假如快指针步长为2,慢指针步长为1,那么当快慢指针同时从首结点出发时,如果在某个时刻两个指针相等,那么链表必定是循环链表。
  • 判断循环链表入口点,当两个指针相等时,此时快指针从首结点出发,步长为1,慢指针继续出发,两个指针必定在循环链表入口点相遇。

有了前面快慢指针的理解那么我们怎么应用到这道题目中来呢,我们首先举个例子说明下情况:

1 3 4 2 2 0 1 2 3 4

上面表格第一行代表数组nums里面的数值,第二行代表数组里面数值的索引。
很明显,假如我们通过以下方式来获取数值:
定义一个变量num,第一次,num = nums[0];第二次num = nums[num];第三次num = nums[num],那么获取的数值变化如下所示:

1 3 2 4 2 4 2 0 1 3 2 4 2 4

很明显,通过每次nums[num]的赋值,数值最后会进入一个循环,说明只要这个数组里存在重复项,那么按照刚刚的赋值方法必定会进入循环。而且我们发现首次进入循环时的数字就是我们的重复项。因此我们可以通过建立快慢指针,找到循环数组的入口点,这个入口点的数值就是本题需要求解的重复项!

代码

class Solution {public:    int findDuplicate(vector<int>& nums) {        if(nums.size()>1)        {            int slow = nums[0],fast=nums[nums[0]];            while(slow!=fast)            {                slow = nums[slow];                fast = nums[nums[fast]];            }            fast =0;            while(slow!=fast)            {                fast = nums[fast];                slow = nums[slow];            }            return slow;        }        return -1;    }};
原创粉丝点击