2021年04月15日 周四 天气大风 【不悲叹过去,不荒废现在,不惧怕未来】
1. 题目简介
2. 题解
2.1 交换法(修改原数组)
题目要求不能修改数组,不然用交换法很容易求解。类似 剑指 Offer 03. 数组中重复的数字,不过代码实现起来更简单。
class Solution {
public:
int findDuplicate(vector<int>& nums) {
while(nums[0]!=nums[nums[0]]){
swap(nums[0],nums[nums[0]]);
}
return nums[0];
}
};
时间复杂度:O(n)
空间复杂度:O(1)
2.2 二分查找法
class Solution {
public:
int findDuplicate(vector<int>& nums) {
int len = nums.size();
int left = 1;
int right = len - 1;
while (left < right) {
int mid = left + (right - left) / 2;
int cnt = 0;
for (int num : nums) {
if (num <= mid) {
cnt += 1;
}
}
// 根据抽屉原理,小于等于 4 的个数如果严格大于 4 个,此时重复元素一定出现在 [1..4] 区间里
if (cnt > mid) {
// 重复元素位于区间 [left..mid]
right = mid;
}
else {
// if 分析正确了以后,else 搜索的区间就是 if 的反面区间 [mid + 1..right]
left = mid + 1;
}
}
return left;
}
};
2.3 龟兔赛跑算法
从0开始建立链表,如数组 [1,3,4,2,2] ,建立的链表为:0->1->3->2->4->2 ,很明显出现了环。
详细可见:剑指offer 23. 链表中环的入口节点(龟兔赛跑算法,数学推理)
class Solution {
public:
int findDuplicate(vector<int>& nums) {
int slow = 0, fast = 0;
slow = nums[slow];
fast = nums[nums[fast]];
while(fast != slow){
slow = nums[slow];
fast = nums[nums[fast]];
}
// 慢指针回到原点
slow = 0;
while(fast != slow){
slow = nums[slow];
fast = nums[fast];
}
return slow;
}
};
时间复杂度:O(n)
空间复杂度:O(1)
参考文献
https://leetcode-cn.com/problems/find-the-duplicate-number/solution/287xun-zhao-zhong-fu-shu-by-kirsche/
https://leetcode-cn.com/problems/find-the-duplicate-number/solution/er-fen-fa-si-lu-ji-dai-ma-python-by-liweiwei1419/