287. Find the Duplicate Number

妖狐艹你老母 2022-06-01 00:36 293阅读 0赞

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.

  1. Note:
  2. - You must not modify the array (assume the array is read only).
  3. - You must use only constant, O(1) extra space.
  4. - Your runtime complexity should be less than O(n2).
  5. - There is only one duplicate number in the array, but it could be repeated more than once.

题目描述:给一个n+1长度的数组,其中的数在1~n中,找到那个重复的元素

要求:不能修改数组
空间复杂度为O(1)
时间复杂度不能超过O(n^2)
数组中的元素可能不止重复一次

思路:其实这道题并不是多难,如果是使用之前的负数标记法,循环一遍数组,将每个元素值对应的数组元素取相反数,然后在之后的循环中进行判断,如果是相同元素值的话,那么所对应的数组元素应该是小于0的,但是这样不符合题目的要求

正是题目中的要求增加难度性,这里主要分析下最优解中的方法,空间复杂度为1,时间复杂度为n。这个方法叫做映射找环法

基本思想:将数组抽象为一条线和一个圆环,因为1~n 之间有n+1个数,所以一定有重复数字出现,所以重复的数字即是圆环与线的交汇点。然后设置两个指针,一个快指针一次走两步,一个慢指针一次走一步。当两个指针第一次相遇时,令快指针回到原点(0)且也变成一次走一步,慢指针则继续前进,再次回合时即是线与圆环的交汇点。为什么一定会汇合?为什么汇合点就是重复数字?不要着急,我们先把Linked list circle问题弄明白。如图1所示,两个指针同时从直线起点开始,假设在x处第一次汇合,xo之间距离为x,那么快指针走过的路程为a+c+x,慢指针走过的路程为a+x,所以a+c+x=2(a+x),所以c=a+x,也就是SO之间的距离等于xo,所以令快指针从起点开始一次一步,慢指针从x开始,同时前进,则必会在O处相遇!
这里写图片描述

明白上面的之后,看下变型之后的,把数组抽象成线和圆环,举例来说,假设我们有一个数组是nums[]=[1,2,3,4,5,5,6,7],pf代表快指针,ps代表慢指针,初始ps指向nums[0],即1,pf指向nums[nums[0]],即2,行动一次后,ps指向nums[1],即2,pf指向nums[nums[2]],即4,再动一次,ps指向nums[2],即3,pf则指向了nums[nums[4]],即5;可以发现pf一旦指向5后便不会再动,因为nums[5]一直为5,直到ps慢慢追上,然后令pf从头开始,ps一直在5处停留,最后定会相遇在这里,而这里就是重复数字。

换种说法,假设数组中没有重复,那我们可以做到这么一点,就是将数组的下标和1到n每一个数一对一的映射起来。比如数组是213,则映射关系为0->2, 1->1, 2->3。假设这个一对一映射关系是一个函数f(n),其中n是下标,f(n)是映射到的数。如果我们从下标为0出发,根据这个函数计算出一个值,以这个值为新的下标,再用这个函数计算,以此类推,直到下标超界。实际上可以产生一个类似链表一样的序列。比如在这个例子中有两个下标的序列,0->2->3。

但如果有重复的话,这中间就会产生多对一的映射,比如数组2131,则映射关系为0->2, {1,3}->1, 2->3。这样,我们推演的序列就一定会有环路了,这里下标的序列是0->2->3->1->1->1->1->…,而环的起点就是重复的数。


代码如下:

  1. public int findDuplicate(int[] nums) {
  2. /*
  3. int rep = 0;
  4. for (int i = 0;i<nums.length; i++){
  5. int var = Math.abs(nums[i]) -1;
  6. if (nums[var] < 0){
  7. rep = Math.abs(var+1);
  8. }
  9. nums[var] = -nums[var];
  10. }
  11. return rep;
  12. */
  13. int slow = 0;
  14. int fast = 0;
  15. do{
  16. slow = nums[slow];
  17. fast = nums[nums[fast]];
  18. } while(slow != fast);
  19. int find = 0;
  20. while(find != slow){
  21. slow = nums[slow];
  22. find = nums[find];
  23. }
  24. return find;
  25. }

发表评论

表情:
评论列表 (有 0 条评论,293人围观)

还没有评论,来说两句吧...

相关阅读