解题思路
设置快指针fptr, 慢指针sptr。由于题设数组每个下标都与值对应,如果通过指针下标访问该地址数值,再将这个地址作为下一次访问的数组下标,那么一定能够遍历完整个数组。由于数组中存在重复数值,一定会进入一个环。
sptr = nums[sptr]; //慢指针,指向当前地址数值
fptr = nums[nums[fptr]]; //快指针,指向与当前地址数值相等的数组下标所对应的值
快指针先进入环,慢指针后进入,且两指针一定相遇。设n为环的长度,设慢指针走过的路程为k*n,(k为任意正整数)则:
由,
;
两指针相遇时,它们在环上位置是一样的,那么两个指针之间的路程差就只能是k个环的长度
fptr在环内位置为,那么只需要让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;
}