当前位置: 首页 > news >正文

【LeetCode 热题 100】215. 数组中的第K个最大元素(Python 快速选择详解)

在刷 LeetCode 的过程中,“第K大”是一个非常高频的考点,而题目 215. 数组中的第K个最大元素 就是经典代表。这道题不仅考察我们对排序的理解,还挑战我们写出时间复杂度为 O(n) 的算法。

本文将带你深入理解并实现一个基于快速选择(Quickselect)的高性能解法。


🧩 题目描述

给定一个整数数组 nums 和一个整数 k,请返回数组中第 k 个最大的元素。

⚠️ 注意:题目中要求是第 k 大,而不是第 k 个不同的元素。

示例

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

❗️暴力解法(不推荐)

最简单的方法就是对数组排序后取倒数第 k 个元素:

def findKthLargest(nums, k):nums.sort()return nums[-k]

虽然代码简洁,但排序的时间复杂度是 O(n log n),不符合题目期望的 O(n) 要求。


💡 正确姿势:快速选择算法(Quickselect)

📚 Quickselect 是什么?

快速选择是快速排序的“变种”,它利用了分治的思想,在每次划分时只处理可能包含答案的那一半,从而平均时间复杂度降为 O(n)。


🧠 算法思路

  1. 随机选一个 pivot(基准元素)

  2. 将数组划分为三部分:

    • big:所有大于 pivot 的数
    • equal:所有等于 pivot 的数
    • small:所有小于 pivot 的数
  3. 判断第 k 大数在哪个部分:

    • 如果在 big 中,递归查找 big 中的第 k
    • 如果在 equal 中,直接返回 pivot
    • 如果在 small 中,调整 k,递归查找 small 中的 (k - big数 - equal数)

✅ 完整 Python 实现

import randomclass Solution:def findKthLargest(self, nums, k):def quick_select(nums, k):pivot = random.choice(nums)big = [num for num in nums if num > pivot]small = [num for num in nums if num < pivot]equal_count = len(nums) - len(big) - len(small)if k <= len(big):return quick_select(big, k)elif k <= len(big) + equal_count:return pivotelse:return quick_select(small, k - len(big) - equal_count)return quick_select(nums, k)

🔍 递归是怎么用的?

  • 递归本质:自己调用自己,逐步缩小问题范围。
  • 每次我们都在更小的数组中递归地寻找第 k 大元素。
  • 有效缩小搜索空间,每次最多处理一半元素。

🎯 举个例子

nums = [3,2,1,5,6,4], k = 2 为例:

  • 首轮 pivot 可能是 3:

    • big = [5, 6, 4],长度为 3
  • 因为 2 ≤ 3,说明我们要找的第 2 大在 big

  • 递归进入 quick_select([5, 6, 4], 2)

  • 再选 pivot,比如是 5:

    • big = [6]equal = [5]small = [4]
  • 此时 2 落在 equal 区间,返回 5,找到答案!


⚙️ 时间 & 空间复杂度分析

项目分析
时间复杂度平均 O(n),最坏 O(n²)(极少发生)
空间复杂度O(n)(由于切片产生的新列表)

🧠 优化建议

  1. 节省空间:可以用原地 partition(如 Lomuto 法)替代切片,避免生成新列表。
  2. 面试场景:如果面试官没要求 O(n),直接用排序即可快速出解。
  3. 重复元素处理:等于 pivot 的元素不能漏掉,需要单独统计。

📌 总结

  • 快速选择是解决“第K大”、“第K小”类问题的利器
  • 随机选择 pivot 是算法性能稳定的关键
  • 递归思想+分治策略,让问题逐步缩小,最终得到答案

📁 推荐刷题练习

  • LeetCode 215. 数组中的第K个最大元素(本文)
  • LeetCode 347. 前 K 个高频元素(用堆)
  • LeetCode 703. 数据流中的第 K 大元素(实时处理)

希望本文对你理解快速选择算法和递归有实质性的帮助!如果觉得有用,欢迎点赞、收藏、评论支持 🙌

相关文章:

  • 探索C++内存管理
  • UE5 PCG学习笔记
  • 初等数论--莫比乌斯反演
  • 【数据结构入门训练DAY-30】数的划分
  • STM32F103单片机在不需要使用 JTAG 调试接口的情况下,释放引脚给其他功能使用。
  • React Hooks 精要:从入门到精通的进阶之路
  • TWAS、GWAS、FUSION
  • 【读书笔记】《编码:隐匿在计算机软硬件背后的语言》01 逻辑与开关
  • 【单片机毕业设计16-基于stm32c8t6的智慧交通灯系统设计】
  • 湖南(源点咨询)市场调研 商业综合体定位调研分享(下篇)
  • Spring Boot 中如何启用 MongoDB 事务
  • 华为云服务器核心用途全景解析:从基础服务到行业革新​​
  • # 深度学习实操 附录B 深入解析 tensorflow 自动微分
  • 【Python】装饰器在装什么
  • 递归函数(斐波那契数列0,1,1,2,3,5,8,13,21,34,55...)
  • 运放OP方向技术要点和大厂题目解析
  • LWIP的IP协议笔记
  • Java中的代理机制
  • STC32G12K128程序大于64KB使用128KB用户EEPROM设置
  • JavaWeb基础
  • 四川资阳市原市长王善平被双开,“笃信风水,大搞迷信活动”
  • 洲际酒店:今年第一季度全球酒店平均客房收入同比增长3.3%
  • 两部门发布外汇领域行刑反向衔接案例,织密金融安全“防护网”
  • 计划招录2577人,“国考”补录8日开始报名
  • 巴方称印军发动24起袭击,巴境内6处地点遭袭致8人死亡
  • 中标多家学校采购项目的App查成绩需付费?涉事公司回应