首页 > 其他分享 >287. Find the Duplicate Number

287. Find the Duplicate Number

时间:2023-03-07 13:00:59浏览次数:40  
标签:快慢 slow nums fast Duplicate num Find 287 指针

##题目
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 n+1 n+1的数组,数组里面的值的值域为 [ 1 , n ] [1,n] [1,n];那么请找出数组里面的重复项。这道题目如果仔细分析的话,本质上就是个快慢指针的问题。首先什么是快慢指针呢?
###快慢指针
快慢指针其实是计算机系统原理里的知识点,快慢指针中的快慢指的是移动的步长,即每次向前移动速度的快慢。例如可以让快指针每次沿链表向前移动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;
        
    }
};

标签:快慢,slow,nums,fast,Duplicate,num,Find,287,指针
From: https://blog.51cto.com/u_15996214/6105934

相关文章

  • find命令
    查找多种后缀文件:find<path>[-name<pattern>][-or-name<pattern>]以查找所有.c和.cpp为例:ubuntu@VM-24-13-ubuntu:~/apps/ddns-serv/server$find../-name"*......
  • find
    find通过文件(名称)、属性、时间、所有者查找文件option参数参数说明-name搜索文件名,支持正则-i忽略大小写搜索,默认对大小写敏感-size大小写搜索,大于(+......
  • Error: Cannot find module ‘webpack-cli/package.json‘
    webpack安装后需要安装webpackcli:npminstall-dwebpack-cli1运行后报错:Error:Cannotfindmodule'webpack-cli/package.json'1解决方案:全局安装webpack-clinpmi......
  • flutter doctor错误:“Unable to find bundled Java version.”
    实际是找不到/Applications/AndroidStudio.app/Contents/jre/jdk执行如下命令:cd/Applications/AndroidStudio.app/Contents/ln-sjbrjrecdjreln-sContents......
  • Unable to find real location for: <frozen codecs>
    问题描述pycharm在debug时出现了如下bug-------------------------------------------------------------------------------pydevdebugger:CRITICALWARNING:This......
  • Windows上使用CMake GUI编译开源代码时,提示:cmake Could NOT find ZLIB (missing:ZLIB_
    有的时候就算在CMakeGUI中配置完ZLIB_LIBRARY和PNG_LIBRARY和PNG_PNG_INCLUDE_DIR等相关路径,还是提示上述错误。原因还是由于编译某源码时遗漏了对第三方开源依赖库的配置......
  • 学习笔记287—为什么要开发 Go 这门新语言?有什么优势?
    编程语言已经非常多,偏性能敏感的编译型语言有C、C++、Java、C#、Delphi和Objective-C等,偏快速业务开发的动态解析型语言有PHP、Python、Perl、Ruby、JavaScript和Lua等,面......
  • BZOJ 2870 最长道路
    BZOJ2870最长道路题意给定一棵\(n\)个节点的树,求树上一条链,使得链的长度乘链上所有点中最小权值所得的积最大\(n\le5\times10^4\)链长度是链上点的个数题面做......
  • unable to find valid certification path to requested target
    摘要:记录一下,防止忘记。当接口调用https接口的时候,需要证书授权认证 一、下载证书(需要下载cer格式的) 导出选择这时候生成一个der格式的证书,别慌,再双击这个der文......
  • E - Find Permutation
    E-FindPermutationhttps://atcoder.jp/contests/abc291/tasks/abc291_e 思路对于能唯一确定的情况,必然存在一个升序路径AX1<AX2<....<AXn***如果有连个......