【数组基础】Leetcode 287. 寻找重复数

题目链接

参考视频

解题思路

        设置快指针fptr, 慢指针sptr。由于题设数组每个下标都与值对应,如果通过指针下标访问该地址数值,再将这个地址作为下一次访问的数组下标,那么一定能够遍历完整个数组。由于数组中存在重复数值,一定会进入一个环。

sptr = nums[sptr];        //慢指针,指向当前地址数值
fptr = nums[nums[fptr]];  //快指针,指向与当前地址数值相等的数组下标所对应的值

快指针先进入环,慢指针后进入,且两指针一定相遇。设n为环的长度,设慢指针走过的路程为k*n,(k为任意正整数)则:
fptr=2*slowfptr = 2kn;

两指针相遇时,它们在环上位置是一样的,那么两个指针之间的路程差就只能是k个环的长度
fptr在环内位置为y = 2kn-t-(k-1)n=kn-t,那么只需要让fast再走t,就能fast在环内位置为y=kn,即到达入口。
而如果此时重置sptr为0,继续循环,则当sptr在入口时,sptr = t,fptr = kn(注意这一次循环fptr步长和sptr保持一致),刚好也在入口。

sptr = 0;
while(sptr != fptr)
{
    sptr = nums[sptr];
    fptr = nums[fptr];
}
return sptr;

环的入口出即为重复的数字,输出该数字即可。

完整代码

#include <iostream>
#include <vector>

using namespace std;

class Solution {
public:
    int findDuplicate(vector<int>& nums)
    {
        int sptr = 0;
        int fptr = 0;
        goto start;

        while(sptr != fptr)
        {
start:
            sptr = nums[sptr];
            fptr = nums[nums[fptr]];
        }

        sptr = 0;
        while(sptr != fptr)
        {
            sptr = nums[sptr];
            fptr = nums[fptr];
        }
        return sptr;
    }
};


int main()
{
    vector<int> nums = {2,5,9,6,9,3,8,9,7,1};
    int output;
    Solution sol;
    output = sol.findDuplicate(nums);
    cout << output << endl;
    return 0;
}
  • 7
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
1. 二分法 5 1.1. 什么是二分查找 5 1.2. 如何识别二分法 5 1.3. 二分法模板 6 1.3.1. 模板一 6 1.3.1.1. 模板代码 6 1.3.1.2. 关键属性 7 1.3.1.3. 语法说明 7 1.3.1.4. Lc69:x的平方根 8 1.3.1.5. Lc374:猜数大小 9 1.3.1.6. Lc33:搜索旋转数组 11 1.3.2. 模板二 13 1.3.2.1. 模板代码 13 1.3.2.2. 关键属性 14 1.3.2.3. 语法说明 14 1.3.2.4. Lc278:第一个错误版本 14 1.3.2.5. Lc162:寻找峰值 16 1.3.2.6. Lc153:寻找旋转排序数组最小值 19 1.3.2.7. Lc154:寻找旋转排序数组最小值II 20 1.3.3. 模板三 22 1.3.3.1. 模板代码 22 1.3.3.2. 关键属性 23 1.3.3.3. 语法说明 23 1.3.3.4. LC-34:在排序数组中查找元素的第一个和最后一个 23 1.3.3.5. LC-658:找到K个最接近的元素 25 1.3.4. 小结 28 1.4. LeetCode中二分查找题目 29 2. 双指针 30 2.1. 快慢指针 31 2.1.1. 什么是快慢指针 31 2.1.2. 快慢指针模板 31 2.1.3. 快慢指针相关题目 32 2.1.3.1. LC-141:链表是否有环 32 2.1.3.2. LC-142:环形链表入口 34 2.1.3.3. LC-876:链表的中间节点 37 2.1.3.4. LC-287:寻找复数 40 2.2. 滑动窗口 43 2.2.1. 什么是滑动窗口 43 2.1.4. 常见题型 44 2.1.5. 注意事项 45 2.1.6. 滑动窗口模板 45 2.1.7. 滑动窗口相关题目 46 2.1.7.1. LC-3:无复字符的最长子串 47 2.1.7.2. LC-76:最小覆盖子串 49 2.1.7.3. LC-209:长度最小的子数组 54 2.1.7.4. LC-239:滑动窗口最大值 57 2.1.7.5. LC-395:至少有K个复字符的最长子串 60 2.1.7.6. LC-567:字符串排列 62 2.1.7.7. LC-904:水果成篮 64 2.1.7.8. LC-424:替换后的最长复字符 66 2.1.7.9. LC-713:乘积小于K的子数组 67 2.1.7.10. LC-992:K个不同整数的子数组 70 2.3. 左右指针 73 2.3.1. 模板 73 2.3.2. 相关题目 73 2.3.2.1. LC-76:删除倒数第N个节点 74 2.3.2.2. LC-61:旋转链表 76 2.3.2.3. LC-80:删除有序数组中的复项 79 2.3.2.4. LC-86:分割链表 80 2.3.2.5. LC-438:找到字符串中所有字母的异位词 82 3. 模板 85 2.3.2.6. LC-76:删除倒数第N个节点 85

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Andy_Xie007

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值