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),只使用了常数个变量。