LeetCode 287. 寻找重复数(龟兔赛跑;二分查找)

2021年04月15日 周四 天气大风 【不悲叹过去,不荒废现在,不惧怕未来】



1. 题目简介

287. 寻找重复数
在这里插入图片描述

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/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值