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

【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 大元素(实时处理)

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

http://www.xdnf.cn/news/5248.html

相关文章:

  • 高精度加减
  • 普通IT的股票交易成长史--股价起伏的真相-缺口(2)
  • 2010-2020年 分省工业品月度产量数据-社科数据
  • 分析AMD业绩突飞猛进的原因
  • [ctfshow web入门] web71
  • SpringBoot项目容器化进行部署,meven的docker插件远程构建docker镜像
  • gvm安装go报错ERROR: Failed to use installed version
  • Linux进程信号的捕捉处理方式
  • 【Java学习】枚举(匿名类详解)
  • 基于大模型的新型隐球菌脑膜炎智能诊疗全流程系统设计与实现的技术方案文档
  • CD37.【C++ Dev】string类的模拟实现(上)
  • fastmcp: 更好用的 MCP Python 框架
  • SlideLoss与FocalLoss在YOLOv8分类损失中的应用及性能分析
  • 指针运算典型例题解析
  • IOC和Bean
  • 【读书笔记】《编码:隐匿在计算机软硬件背后的语言》01 逻辑与开关
  • Android方法耗时监控插件开发
  • Java 基础面试题
  • 自定义类型-结构体(一)
  • 【Rust】枚举和模式匹配
  • 2025年数维杯赛题C题专家 组委会C题专家疑集锦
  • 5.8线性动态规划2
  • SpringMVC-执行流程
  • 40、C# 数组、链表、哈希、队列、栈数据结构的特点、优点和缺点
  • AI:生成对抗网络(GAN)
  • 【Vue】vuex的getters mapState mapGetters mapMutations mapActions的使用
  • yocto的大致工作流程
  • CSS渲染性能优化
  • MySQL进阶篇2_SQL优化、锁
  • 探索 JWT(JSON Web Token):原理、结构与实践应用对比