剑指Offer03 -- 数组中重复的数字 哈希表

0 题目描述

leetcode原题链接:剑指 Offer 03. 数组中重复的数字
在这里插入图片描述

1 排序查找

数组排序之后,重复元素一定是相邻的,然后前后两两比较,如果有重复的直接返回。

class Solution:
    def findRepeatNumber(self, nums: List[int]) -> int:
        if not nums or len(nums)<=1: return -1
        nums.sort()
        for i in range(len(nums)-1):
            if nums[i] == nums[i+1]:
                return nums[i]
        return -1

复杂度分析
时间复杂度: O ( n l g n ) O(nlgn) O(nlgn),使用了排序算法。
空间复杂度: O ( 1 ) O(1) O(1),只使用了常数个变量。

2 哈希算法

利用哈希表来解决问题,从头到尾按顺序扫描数组的每个数字,每扫描到一个数字的时候,都可以用 O ( 1 ) O(1) O(1)的时间来判断哈希表是否已经包含了该数字。如果哈希表里还没有这个数字,就把它加入哈希表。如果哈希表里已经存在该数字,就找到一个重复的数字。

class Solution:
    def findRepeatNumber(self, nums: List[int]) -> int:
        # from collections import defaultdict
        if not nums or len(nums)<=1: return -1
        dicts = collections.defaultdict(int)
        for num in nums:
            dicts[num] += 1
            if dicts[num] == 2:
                return num
        return -1

复杂度分析
时间复杂度: O ( n ) O(n) O(n), 遍历数组。
空间复杂度: O ( n ) O(n) O(n),需要一个大小为 O ( n ) O(n) O(n)的哈希表。

3 数组哈希表

  • 由于数组元素的值都在指定的范围内,这个范围恰恰好与数组的下标可以一一对应;
  • 因此看到数值,就可以知道它应该放在什么位置,这里数字nums[i] 应该放在下标为 i 的位置上,这就像是我们人为编写了哈希函数,这个哈希函数的规则还特别简单;
  • 而找到重复的数就是发生了哈希冲突;
    在这里插入图片描述

类似问题还有「力扣」第 41 题: 缺失的第一个正数、「力扣」第 442 题: 数组中重复的数据、「力扣」第 448 题: 找到所有数组中消失的数字 。

class Solution:
    def findRepeatNumber(self, nums: List[int]) -> int:
        n = len(nums)
        if not nums or n <= 1:
            return -1
        for i in range(n):
            if nums[i] < 0 or nums[i] > n-1:
                return -1
        for i in range(n):
            while nums[i]!=i:
                if nums[i] == nums[nums[i]]:
                    return nums[i]
                temp = nums[i]
                nums[i] = nums[temp]
                nums[temp] = temp
        return -1

复杂度分析
时间复杂度: O ( n ) O(n) O(n),遍历数组。
空间复杂度: O ( 1 ) O(1) O(1),只使用了常数个变量。

参考资料

把数组视为哈希表,找到重复的数就是发生了哈希冲突
4种解决方式(set,排序,临时数组,放到指定位置)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值