【数组基础】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;
}
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Andy_Xie007

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

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

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

打赏作者

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

抵扣说明:

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

余额充值