217. 存在重复元素

给定一个整数数组,判断是否存在重复元素。

如果任何值在数组中出现至少两次,函数返回 true。如果数组中每个元素都不相同,则返回 false。

示例 1:

输入: [1,2,3,1]
输出: true
示例 2:

输入: [1,2,3,4]
输出: false
示例 3:

输入: [1,1,1,3,3,4,3,2,4,2]
输出: true

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/contains-duplicate
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路:
排序, 先排序,排序后重复的元素在一起 若相邻的元素相等 则返回true.

哈希表. 使用hashSet

暴力.遍历每一种情况 相等就返回true.

方法一:朴素线性查找 【超时】
直觉

对于一个有 nn 个整数的数组,一共有 C(n,2) = \frac{n(n+1)}{2}C(n,2)=
2
n(n+1)

对整数。因此,我们可以对所有的 \frac{n(n+1)}{2}
2
n(n+1)

对进行检测,看它们是否相同。

算法

为了实现这个思路,我们使用线性查找算法,这是最简单的查找算法。线性查找是一种检查特定值是否在列表中的算法,做法是依次逐个检查列表中的元素,直到找到满足的元素。

对于本问题,我们循环遍历全部 nn 个数。对于第 ii 个整数 nums[i],我们对前 i-1 个整数查找 nums[i] 的重复值。若找到,则返回 True; 否则继续。在程序最后,返回 False。

为了证明算法的正确性,我们定义了循环不变式。循环不变式是指在每次迭代前和后均保持不变的性质。了解循环不变式对理解循环的意义十分重要。下面就是循环不变式:

在下一次搜索之前,搜索过的整数中没有重复的整数。

循环不变式在循环之前为真,因为还没有搜索过的整数。每次循环,我们查找当前元素的任何可能重复。如果发现重复项,则函数返回 True 退出;如果没有发现,则不变式仍然成立。

因此,如果循环结束,循环不变式说明全部 nn 个整数中不存在重复元素。

Java
public boolean containsDuplicate(int[] nums) {
for (int i = 0; i < nums.length; ++i) {
for (int j = 0; j < i; ++j) {
if (nums[j] == nums[i]) return true;
}
}
return false;
}
// Time Limit Exceeded
复杂度分析

时间复杂度 : O(n^2)O(n
2
)。最坏的情况下,需要检查 \frac{n(n+1)}{2}
2
n(n+1)

对整数。因此,时间复杂度为 O(n^2)O(n
2
)。

空间复杂度 : O(1)O(1)。只使用了常数额外空间。

注意

本方法在 Leetcode 上会超时。一般而言,如果一个算法的时间复杂度为 O(n^2)O(n
2
),它最多能处理 nn 大约为 10^410
4
的数据。当 nn 接近 10^510
5
时就会超时。

方法二:排序 【通过】
直觉

如果存在重复元素,排序后它们应该相邻。

算法

本方法使用排序算法。由于比较排序算法,如堆排序,可以在最坏情况下具有 O(n \log n)O(nlogn) 的时间复杂度。因此,排序经常是很好的预处理方法。排序之后,我们可以扫描已排序的数组,以查找是否有任何连续的重复元素。

Java
public boolean containsDuplicate(int[] nums) {
Arrays.sort(nums);
for (int i = 0; i < nums.length - 1; ++i) {
if (nums[i] == nums[i + 1]) return true;
}
return false;
}
复杂度分析

时间复杂度 : O(n \log n)O(nlogn)。
排序的复杂度是 O(n \log n)O(nlogn),扫描的复杂度是 O(n)O(n)。整个算法主要由排序过程决定,因此是 O(n \log n)O(nlogn)。

空间复杂度 : O(1)O(1)。
这取决于具体的排序算法实现,通常而言,使用 堆排序 的话,是 O(1)O(1)。

注意

此处的算法实现对原始数组进行排序,修改了原始数组。通常,除非调用方清楚输入数据将被修改,否则不应该随意修改输入数据。可以先复制 nums,然后对副本进行操作。

方法三:哈希表 【通过】
直觉

利用支持快速搜索和插入操作的动态数据结构。

算法

从方法一中我们知道,对无序数组的查找操作的时间复杂度为 O(n)O(n),而我们会重复调用查找操作。因此,使用搜索时间更快的数据结构将加快整个算法的速度。

有许多数据结构常用作动态集合,如二进制搜索树和哈希表。这里我们需要的操作是 search 和 insert。对于平衡二叉搜索树(Java 中的 TreeSet 或 TreeMap),search 和 insert 的时间复杂度均为 O(\log n)O(logn)。对于哈希表(Java 中的 HashSet 或 HashMap),search 和 insert 的平均时间复杂度为 O(1)O(1)。因此,通过使用哈希表,我们可以达到在线性时间复杂度解决问题。

Java
public boolean containsDuplicate(int[] nums) {
Set set = new HashSet<>(nums.length);
for (int x: nums) {
if (set.contains(x)) return true;
set.add(x);
}
return false;
}
复杂度分析

时间复杂度 : O(n)O(n)。
search() 和 insert() 各自使用 nn 次,每个操作耗费常数时间。

空间复杂度 : O(n)O(n)。哈希表占用的空间与元素数量是线性关系。

注意

对于一些特定的 nn 不太大的测试样例,本方法的运行速度可能会比方法二更慢。这是因为哈希表在维护其属性时有一些开销。要注意,程序的实际运行表现和 Big-O 符号表示可能有所不同。Big-O 只是告诉我们在 充分 大的输入下,算法的相对快慢。因此,在 nn 不够大的情况下, O(n)O(n) 的算法也可以比 O(n \log n)O(nlogn)的更慢。

作者:LeetCode
链接:https://leetcode-cn.com/problems/contains-duplicate/solution/cun-zai-zhong-fu-yuan-su-by-leetcode/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值