【刷穿 LeetCode】414. 第三大的数 : 「排序」&「遍历」

简介: 【刷穿 LeetCode】414. 第三大的数 : 「排序」&「遍历」

网络异常,图片无法展示
|


题目描述



这是 LeetCode 上的 414. 第三大的数 ,难度为 简单


Tag : 「排序」、「数组」、「模拟」


给你一个非空数组,返回此数组中 第三大的数 。如果不存在,则返回数组中最大的数。


示例 1:


输入:[3, 2, 1]
输出:1
解释:第三大的数是 1 。
复制代码


示例 2:


输入:[1, 2]
输出:2
解释:第三大的数不存在, 所以返回最大的数 2 。
复制代码


示例 3:


输入:[2, 2, 3, 1]
输出:1
解释:注意,要求返回第三大的数,是指在所有不同数字中排第三大的数。
此例中存在两个值为 2 的数,它们都排第二。在所有不同数字中排第三大的数为 1 。
复制代码


提示:


  • 1 <= nums.length <= 10^4104
  • -2^{31}231 <= nums[i] <= 2^{31} - 12311


进阶:你能设计一个时间复杂度 O(n) 的解决方案吗?


Set 去重 + 排序



题目要求返回含重复元素的数组 numsnums 中的第三大数。


一个朴素的做法是,先使用 Set 对重复元素进行去重,然后对去重后的元素进行排序,并返回第三大的元素。


代码:


class Solution {
    public int thirdMax(int[] nums) {
        Set<Integer> set = new HashSet<>();
        for (int x : nums) set.add(x);
        List<Integer> list = new ArrayList<>(set);
        Collections.sort(list);
        return list.size() < 3 ? list.get(list.size() - 1) : list.get(list.size() - 3); 
    }
}
复制代码


  • 时间复杂度:使用 Set 去重的复杂度为 O(n)O(n);排序复杂度为 O(n\log{n})O(nlogn)。整体复杂度为 O(n\log{n})O(nlogn)
  • 空间复杂度:O(n)O(n)


有限变量 + 遍历



经典的找数组次大值的做法是使用两个变量 ab 分别存储遍历过程中的最大值和次大值。


假设当前遍历到的元素为 xx,当满足如下条件时,考虑更新 a 或者 b:


  1. x > ax>a 时,说明最大值被更新,同时原来的最大值沦为次大值。即有 b = a; a = x;b=a;a=x;
  2. 在条件11不满足,且有x > bx>b时,此时可以根据是否有「严格次大值」的要求,而决定是否要增加x < ax<a的条件:
  • 不要求为「严格次大值」:直接使用 xx 来更新 b,即有 b = xb=x
  • 当要求为「严格次大值」: 此时需要满足 x < ax<a 的条件,才能更新 b


回到本题,同理我们可以使用 abc 三个变量来代指「最大值」、「严格次大值」和「严格第三大值」。


从前往后遍历 numsnums,假设当前元素为 xx,对是否更新三者进行分情况讨论(判断优先级从上往下):


  1. x > ax>a,说明最大值被更新,将原本的「最大值」和「次大值」往后顺延为「次大值」和「第三大值」,并用 xx 更新 a
  2. x < ax<ax > bx>b,说明次大值被更新,将原本的「次大值」往后顺延为「第三大值」,并用 xx 更新 b
  3. x < bx<bx > cx>c,说明第三大值被更新,使用 xx 更新 c


起始时,我们希望使用一个足够小的数来初始化 abc,但由于 num[i]num[i] 的范围为 [-2^{31}, 2^{31} - 1][231,2311],因此需要使用 long 来进行代替。


返回时,通过判断第三大值是否为初始化时的负无穷,来得知是否存在第三大值。


代码:


class Solution {
    long INF = (long)-1e18;
    public int thirdMax(int[] nums) {
        long a = INF, b = INF, c = INF;
        for (int x : nums) {
            if (x > a) {
                c = b; b = a; a = x;
            } else if (x < a && x > b) {
                c = b; b = x;
            } else if (x < b && x > c) {
                c = x;
            }
        }
        return c != INF ? (int)c : (int)a;
    }
}
复制代码


  • 时间复杂度:O(n)O(n)
  • 空间复杂度:O(1)O(1)


最后



这是我们「刷穿 LeetCode」系列文章的第 No.414 篇,系列开始于 2021/01/01,截止于起始日 LeetCode 上共有 1916 道题目,部分是有锁题,我们将先把所有不带锁的题目刷完。


在这个系列文章里面,除了讲解解题思路以外,还会尽可能给出最为简洁的代码。如果涉及通解还会相应的代码模板。


为了方便各位同学能够电脑上进行调试和提交代码,我建立了相关的仓库:github.com/SharingSour…


在仓库地址里,你可以看到系列文章的题解链接、系列文章的相应代码、LeetCode 原题链接和其他优选题解。

相关文章
|
3月前
【LeetCode-每日一题】 删除排序数组中的重复项
【LeetCode-每日一题】 删除排序数组中的重复项
30 4
|
3月前
|
索引
Leetcode第三十三题(搜索旋转排序数组)
这篇文章介绍了解决LeetCode第33题“搜索旋转排序数组”的方法,该问题要求在旋转过的升序数组中找到给定目标值的索引,如果存在则返回索引,否则返回-1,文章提供了一个时间复杂度为O(logn)的二分搜索算法实现。
30 0
Leetcode第三十三题(搜索旋转排序数组)
|
3月前
【LeetCode 37】106.从中序与后序遍历构造二叉树
【LeetCode 37】106.从中序与后序遍历构造二叉树
29 0
|
5月前
|
存储 算法
LeetCode第83题删除排序链表中的重复元素
文章介绍了LeetCode第83题"删除排序链表中的重复元素"的解法,使用双指针技术在原链表上原地删除重复元素,提供了一种时间和空间效率都较高的解决方案。
LeetCode第83题删除排序链表中的重复元素
|
5月前
|
算法 Java
LeetCode第94题二叉树的中序遍历
文章介绍了LeetCode第94题"二叉树的中序遍历"的解法,使用递归实现了中序遍历的过程,遵循了"左根右"的遍历顺序,并提供了清晰的Java代码实现。
LeetCode第94题二叉树的中序遍历
|
5月前
|
算法 索引
LeetCode第34题在排序数组中查找元素的第一个和最后一个位置
这篇文章介绍了LeetCode第34题"在排序数组中查找元素的第一个和最后一个位置"的解题方法,通过使用双指针法从数组两端向中间同时查找目标值,有效地找到了目标值的首次和最后一次出现的索引位置。
LeetCode第34题在排序数组中查找元素的第一个和最后一个位置
|
5月前
|
算法 Python
【Leetcode刷题Python】剑指 Offer 33. 二叉搜索树的后序遍历序列
本文提供了一种Python算法,用以判断给定整数数组是否为某二叉搜索树的后序遍历结果,通过识别根节点并递归验证左右子树的值是否满足二叉搜索树的性质。
29 3
|
5月前
|
Python
【Leetcode刷题Python】105. 从前序与中序遍历序列构造二叉树
LeetCode上105号问题"从前序与中序遍历序列构造二叉树"的Python实现,通过递归方法根据前序和中序遍历序列重建二叉树。
37 3
|
5月前
|
Python
【Leetcode刷题Python】145. 二叉树的后序遍历
LeetCode上145号问题"二叉树的后序遍历"的Python实现方法。
28 2
|
5月前
|
Python
【Leetcode刷题Python】144. 二叉树的前序遍历
LeetCode上144号问题"二叉树的前序遍历"的Python实现方法。
27 1