【LeetCode刷题】寻找重复数

给定一个包含 n + 1 个整数的数组 nums ,其数字都在 [1, n] 范围内(包括 1n),可知至少存在一个重复的整数。

假设 nums 只有 一个重复的整数 ,返回 这个重复的数

你设计的解决方案必须 不修改 数组 nums 且只用常量级 O(1) 的额外空间。

示例 1:

复制代码
输入:nums = [1,3,4,2,2]
输出:2

示例 2:

复制代码
输入:nums = [3,1,3,4,2]
输出:3

示例 3 :

复制代码
输入:nums = [3,3,3,3,3]
输出:3

提示:

  • 1 <= n <=
  • nums.length == n + 1
  • 1 <= nums[i] <= n
  • nums只有一个整数 出现 两次或多次 ,其余整数均只出现 一次

算法解析:

  1. 构建链表逻辑 :把数组的索引 看作链表节点,元素值 看作下一个节点的索引(因数组元素范围是[1,n],不会越界)。由于存在重复数,链表会形成 ,重复数就是环的入口节点

  2. 快慢指针找环

    • 慢指针slow每次走 1 步,快指针fast每次走 2 步,最终会在环内某点相遇。
  3. 找环的入口

    • 从数组起点(nums[0])和环内相遇点,同时出发两个指针(每次走 1 步),相遇处即为环的入口,也就是重复的数。

示例验证

nums = [1,3,4,2,2]为例:

  • 链表结构:0 → 1 → 3 → 2 → 4 → 2(环的入口是2)。
  • 快慢指针相遇后,起点指针与慢指针会在2处相遇,返回结果2

Python代码:

python 复制代码
from typing import List


class Solution:
    """
    寻找数组中重复的数字(满足条件:数组长度n+1,元素范围[1,n],仅一个重复数,可能重复多次)
    核心算法:快慢指针(弗洛伊德环检测),时间复杂度O(n),空间复杂度O(1),不修改原数组
    """

    def findDuplicate(self, nums: List[int]) -> int:
        """
        查找数组中重复的数字
        :param nums: 输入数组,长度为n+1,元素范围[1,n],保证有且仅有一个数字重复
        :return: 重复的数字
        """
        # 边界校验:数组长度小于2时无意义(题目保证输入合法,此处为鲁棒性补充)
        if len(nums) < 2:
            raise ValueError("数组长度至少为2")

        # 1. 快慢指针找环内相遇点(慢指针走1步,快指针走2步)
        slow = nums[0]
        fast = nums[0]
        while True:
            slow = nums[slow]  # 慢指针:每次走1步
            fast = nums[nums[fast]]  # 快指针:每次走2步
            if slow == fast:  # 快慢指针相遇,说明存在环,退出循环
                break

        # 2. 找环的入口(重复数就是环的入口)
        # 原理:从数组起点和相遇点同时出发,每次走1步,相遇处即为环入口
        ptr = nums[0]  # 指针1:从数组起点出发
        while ptr != slow:
            ptr = nums[ptr]  # 指针1走1步
            slow = nums[slow]  # 指针2(原慢指针)走1步

        return ptr


# -------------------------- 测试用例 --------------------------
if __name__ == "__main__":
    solution = Solution()

    # 测试用例1:基础情况
    nums1 = [1, 3, 4, 2, 2]
    print(f"测试用例1: {nums1} → 重复数:{solution.findDuplicate(nums1)}")  # 预期输出:2

    # 测试用例2:重复数在开头
    nums2 = [2, 2, 2, 2, 2]
    print(f"测试用例2: {nums2} → 重复数:{solution.findDuplicate(nums2)}")  # 预期输出:2

    # 测试用例3:最小边界(数组长度2)
    nums3 = [1, 1]
    print(f"测试用例3: {nums3} → 重复数:{solution.findDuplicate(nums3)}")  # 预期输出:1

    # 测试用例4:重复数在中间
    nums4 = [3, 1, 3, 4, 2]
    print(f"测试用例4: {nums4} → 重复数:{solution.findDuplicate(nums4)}")  # 预期输出:3

LeetCode提交代码:

python 复制代码
class Solution:
    def findDuplicate(self, nums: List[int]) -> int:
        # 1. 快慢指针找环内相遇点
        slow = nums[0]
        fast = nums[0]
        while True:
            slow = nums[slow]
            fast = nums[nums[fast]]
            if slow == fast:
                break
        
        # 2. 找环的入口(即重复数)
        ptr = nums[0]
        while ptr != slow:
            ptr = nums[ptr]
            slow = nums[slow]
        return ptr

程序运行截图展示

总结

题目要求在长度为n+1的数组中找到唯一重复的数字(元素范围[1,n]),要求不修改数组且使用O(1)空间。通过将数组视为链表(索引为节点,值为下一节点),利用快慢指针检测环:

  1. 找环:快指针(每次2步)与慢指针(每次1步)相遇;
  2. 找入口 :从起点和相遇点同步移动,相遇点即为重复数。
    示例[1,3,4,2,2]中,链表形成环2→4→2,入口2即为解。算法时间复杂度O(n),空间O(1)。Python代码通过双指针实现,已验证边界用例。
相关推荐
哈里谢顿8 小时前
Python 高并发服务限流终极方案:从原理到生产落地(2026 实战指南)
python
NAGNIP17 小时前
轻松搞懂全连接神经网络结构!
人工智能·算法·面试
NAGNIP17 小时前
一文搞懂激活函数!
算法·面试
董董灿是个攻城狮18 小时前
AI 视觉连载7:传统 CV 之高斯滤波实战
算法
用户8356290780511 天前
无需 Office:Python 批量转换 PPT 为图片
后端·python
爱理财的程序媛1 天前
openclaw 盯盘实践
算法
markfeng81 天前
Python+Django+H5+MySQL项目搭建
python·django
GinoWi1 天前
Chapter 2 - Python中的变量和简单的数据类型
python
JordanHaidee1 天前
Python 中 `if x:` 到底在判断什么?
后端·python