LeetCode 287. 寻找重复数

287. 寻找重复数

  

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

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

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

示例 1:

输入:nums = [1,3,4,2,2]
输出:2

示例 2:

输入:nums = [3,1,3,4,2]
输出:3

提示:

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

进阶:

  • 如何证明 nums 中至少存在一个重复的数字?
  • 你可以设计一个线性级时间复杂度 O(n) 的解决方案吗?

思路

二分查找法,其他方法不满足题目所给进阶要求,暂不考虑,参考 B站视频题解

每一次找中间值,考虑到1-n之间的数一定都会出现,如果非重复情况下每个数只会出现一次,故我们判断小于等于中间值元素个数如果超过了中间值则说明重复的数在左半边(正常情况下【不包含重复元素】时的对应数组的左半边),否则去右半边找(正常情况下【不包含重复元素】时的对应数组的右半边)

举例说明:

  • 正常情况下【不包含重复元素】:小于等于midVal值的个数cnt应该是与midVal值恰好相等的

    比如表示范围为[1,n=4]的nums=[1,2,3,4]中,小于等于2的元素一共有2个:1和2
  • 但在实际情况下【包含重复元素】,比如说 nums=[1,3,4,2,2],那么怎么样找出这个重复的2呢?
  • 我们可以先推理出表示范围为[1,n=4],且正常情况下【不包含重复元素】的中间值:midVal = 1 + (4-1)/2 = 2

    接着我们遍历 实际情况下【包含重复元素】的nums=[1,3,4,2,2],发现其中小于等于 midVal=2 的数实际上有3个,分别为 1,2,2,大于正常情况下的2个

    因此可以判断在实际给出的nums=[1,3,4,2,2]中,存在多余且重复的数,导致小于等于 midVal=2 的元素个数为3个(正常情况下应该为2个)

时间复杂度:二分查找 O(nlogn)

空间复杂度:O(1)

// hash表,空间复杂度为O(n),不满足题目进阶要求
// func findDuplicate(nums []int) int {
//     m := make(map[int]int)
//     for i := 0; i < len(nums); i++ {
//         m[nums[i]] += 1

//         if m[nums[i]] >= 2 {
//             return nums[i]
//         }
//     }

//     return -1
// }

// 二分法 :
// 参考B站视频题解:https://www.bilibili.com/video/BV1qg41137Fg/?spm_id_from=333.337.search-card.all.click&vd_source=2c268e25ffa1022b703ae0349e3659e4
// 思路:每一次找中间值,考虑到1-n之间的数一定都会出现,如果非重复情况下每个数只会出现一次,故我们判断小于等于中间值元素个数如果超过了中间值则说明重复的数在左半边,否则去右半边找
/* 
    举例说明
    正常情况下【不包含重复元素】:小于等于midVal值的个数cnt应该是与midVal值恰好相等的
    比如表示范围为[1,n=4]的nums=[1,2,3,4]中,小于等于2的元素一共有2个:1和2

    但在实际情况下【包含重复元素】,比如说 nums=[1,3,4,2,2],那么怎么样找出这个重复的2呢?

    我们可以先推理出表示范围为[1,n=4],且正常情况下【不包含重复元素】的中间值:midVal = 1 + (4-1)/2 = 2
    接着我们遍历 实际情况下【包含重复元素】的nums=[1,3,4,2,2],发现其中小于等于 midVal=2 的数实际上有3个,分别为 1,2,2,大于正常情况下的2个
    因此可以判断在实际给出的nums=[1,3,4,2,2]中,存在多余且重复的数,导致小于等于 midVal=2 的元素个数为3个(正常情况下应该为2个)
*/

// 时间复杂度:二分法 O(nlogn),空间复杂度:O(1)
func findDuplicate(nums []int) int {
    // nums的数字都在 [1, n] 范围内,所以先定位nums中元素的【值】可能出现的左右范围 low和high
    // 数组 nums包含n + 1 个整数,即 n + 1 = len(nums),因此 n = len(nums) - 1
    n := len(nums)
    low, high := 1, n - 1

    for low <= high {
        midVal, cnt := low + (high - low) / 2, 0
        
        // 统计nums中元素在区间为[1, midVal]中的个数,注意这里midVal表示的是 正常情况下【不包含重复元素】的中间值
        for i := 0; i < n; i++ {
            // 统计实际给出的nums中,小于等于midVal值的元素个数,正常情况下【不包含重复元素】应该是cnt刚好等于midVal
            if nums[i] <= midVal {
                cnt++
            }
        }

        // 重复数位于midVal左侧:
        // 发现cnt大于正常情况下【不包含重复元素】对应的元素个数 midVal,并且这个重复数是小于midVal的
        if cnt > midVal {
            high = midVal - 1

        // 重复数位于midVal右侧:
        // 发现cnt小于正常情况下【不包含重复元素】对应的元素个数 midVal,那就说明要找的重复数不在midVal左侧而在右侧
        } else {
            low = midVal + 1
        }
    }

    return low
}

### Python 实现 LeetCode 287 寻找组内的重复 对于在Python中解决寻找组内重复字的问题,可以采用多种方法来处理这个问题。一种常见的高效解法是利用二分查找或者快慢指针的方法。 #### 方法一:二分查找 通过设定左右边界并逐步缩小范围直到找到重复元素的位置。这种方法不需要额外的空间开销,并且能够有效地减少时间复杂度。 ```python def findDuplicate(nums): low = 1 high = len(nums) - 1 while low <= high: mid = (low + high) >> 1 count = sum(num <= mid for num in nums) if count > mid: duplicate = mid high = mid - 1 else: low = mid + 1 return duplicate ``` 此段代码实现了基于二分查找的解决方案[^3]。 #### 方法二:快慢指针(Floyd 判圈算法) 该方法模仿链表检测环路的思想,在不改变输入组的情况下工作。它使用两个指针——一个移动速度快于另一个;当它们相遇时,则说明存在循环即找到了重复项。 ```python def findDuplicate(nums): tortoise = hare = nums[0] # Phase 1: Find the intersection point of two runners. while True: tortoise = nums[tortoise] hare = nums[nums[hare]] if tortoise == hare: break # Phase 2: Find the entrance to cycle (duplicate element). tortoise = nums[0] while tortoise != hare: tortoise = nums[tortoise] hare = nums[hare] return hare ``` 上述代码展示了如何应用 Floyd 的判圈算法来定位重复值。 这两种方式都能够在 O(n log n) 或者更优的时间复杂度下完成任务,并且不会修改原始据结构。如果考虑空间效率的话,建议优先尝试这些原地工作的算法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值