0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

215.数组中的第K个最大元素(Medium)

算法与数据结构 来源:labuladong 作者:labuladong 2020-12-02 16:00 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

读完本文,可以去力扣解决如下题目:

215.数组中的第 K 个最大元素(Medium)

快速选择算法是一个非常经典的算法,和快速排序算法是亲兄弟。

原始题目很简单,给你输入一个无序的数组nums和一个正整数k,让你计算nums中第k大的元素。

那你肯定说,给nums数组排个序,然后取第k个元素,也就是nums[k-1],不就行了吗?

当然可以,但是排序时间复杂度是O(NlogN),其中N表示数组nums的长度。

我们就想要第k大的元素,却给整个数组排序,有点杀鸡用牛刀的感觉,所以这里就有一些小技巧了,可以把时间复杂度降低到O(NlogK)甚至是O(N),下面我们就来具体讲讲。

力扣第 215 题「数组中的第 K 个最大元素」就是一道类似的题目,函数签名如下:

intfindKthLargest(int[]nums,intk);

只不过题目要求找第k个最大的元素,和我们刚才说的第k大的元素在语义上不太一样,题目的意思相当于是把nums数组降序排列,然后返回第k个元素。

比如输入nums = [2,1,5,4], k = 2,算法应该返回 4,因为 4 是nums中第 2 个最大的元素。

这种问题有两种解法,一种是二叉堆(优先队列)的解法,另一种就是标题说到的快速选择算法(Quick Select),我们分别来看。

二叉堆解法

二叉堆的解法比较简单,实际写算法题的时候,推荐大家写这种解法,先直接看代码吧:

二叉堆(优先队列)是比较常见的数据结构,可以认为它会自动排序,我们前文 手把手实现二叉堆数据结构 实现过这种结构,我就默认大家熟悉它的特性了。

看代码应该不难理解,可以把小顶堆pq理解成一个筛子,较大的元素会沉淀下去,较小的元素会浮上来;当堆大小超过k的时候,我们就删掉堆顶的元素,因为这些元素比较小,而我们想要的是前k个最大元素嘛。当nums中的所有元素都过了一遍之后,筛子里面留下的就是最大的k个元素,而堆顶元素是堆中最小的元素,也就是「第k个最大的元素」。

二叉堆插入和删除的时间复杂度和堆中的元素个数有关,在这里我们堆的大小不会超过k,所以插入和删除元素的复杂度是O(logK),再套一层 for 循环,总的时间复杂度就是O(NlogK)。空间复杂度很显然就是二叉堆的大小,为O(K)。

这个解法算是比较简单的吧,代码少也不容易出错,所以说如果笔试面试中出现类似的问题,建议用这种解法。唯一注意的是,Java 的PriorityQueue默认实现是小顶堆,有的语言的优先队列可能默认是大顶堆,可能需要做一些调整。

快速选择算法

快速选择算法比较巧妙,时间复杂度更低,是快速排序的简化版,一定要熟悉思路。

我们先从快速排序讲起。

快速排序的逻辑是,若要对nums[lo..hi]进行排序,我们先找一个分界点p,通过交换元素使得nums[lo..p-1]都小于等于nums[p],且nums[p+1..hi]都大于nums[p],然后递归地去nums[lo..p-1]和nums[p+1..hi]中寻找新的分界点,最后整个数组就被排序了。

快速排序的代码如下:

关键就在于这个分界点索引p的确定,我们画个图看下partition函数有什么功效:

索引p左侧的元素都比nums[p]小,右侧的元素都比nums[p]大,意味着这个元素已经放到了正确的位置上,回顾快速排序的逻辑,递归调用会把nums[p]之外的元素也都放到正确的位置上,从而实现整个数组排序,这就是快速排序的核心逻辑。

那么这个partition函数如何实现的呢?看下代码:

熟悉快速排序逻辑的读者应该可以理解这段代码的含义了,这个partition函数细节较多,上述代码参考《算法4》,是众多写法中最漂亮简洁的一种,所以建议背住,这里就不展开解释了。

好了,对于快速排序的探讨到此结束,我们回到一开始的问题,寻找第k大的元素,和快速排序有什么关系?

注意这段代码:

intp=partition(nums,lo,hi);

我们刚说了,partition函数会将nums[p]排到正确的位置,使得nums[lo..p-1] < nums[p] < nums[p+1..hi]。

那么我们可以把p和k进行比较,如果p < k说明第k大的元素在nums[p+1..hi]中,如果p > k说明第k大的元素在nums[lo..p-1]中。

所以我们可以复用partition函数来实现这道题目,不过在这之前还是要做一下索引转化:

题目要求的是「第k个最大元素」,这个元素其实就是nums升序排序后「索引」为len(nums) - k的这个元素。

这样就可以写出解法代码:

这个代码框架其实非常像我们前文二分搜索框架的代码,这也是这个算法高效的原因,但是时间复杂度为什么是O(N)呢?按理说类似二分搜索的逻辑,时间复杂度应该一定会出现对数才对呀?

其实这个O(N)的时间复杂度是个均摊复杂度,因为我们的partition函数中需要利用双指针技巧遍历nums[lo..hi],那么总共遍历了多少元素呢?

最好情况下,每次p都恰好是正中间(lo + hi) / 2,那么遍历的元素总数就是:

N + N/2 + N/4 + N/8 + … + 1

这就是等比数列求和公式嘛,求个极限就等于2N,所以遍历元素个数为2N,时间复杂度为O(N)。

但我们其实不能保证每次p都是正中间的索引的,最坏情况下p一直都是lo + 1或者一直都是hi - 1,遍历的元素总数就是:

N + (N - 1) + (N - 2) + … + 1

这就是个等差数列求和,时间复杂度会退化到O(N^2),为了尽可能防止极端情况发生,我们需要在算法开始的时候对nums数组来一次随机打乱:

前文洗牌算法详解写过随机乱置算法,这里就不展开了。当你加上这段代码之后,平均时间复杂度就是O(N)了,提交代码后运行速度大幅提升。

总结一下,快速选择算法就是快速排序的简化版,复用了partition函数,快速定位第 k 大的元素。相当于对数组部分排序而不需要完全排序,从而提高算法效率,将平均时间复杂度降到O(N)。

责任编辑:xj

原文标题:快排亲兄弟:快速选择算法详解

文章出处:【微信公众号:算法与数据结构】欢迎添加关注!文章转载请注明出处。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 算法
    +关注

    关注

    23

    文章

    4810

    浏览量

    98597
  • 代码
    +关注

    关注

    30

    文章

    4977

    浏览量

    74397

原文标题:快排亲兄弟:快速选择算法详解

文章出处:【微信号:TheAlgorithm,微信公众号:算法与数据结构】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    深入解析TPS54372EVM - 215评估模块

    评估模块,了解其性能、特点以及使用过程的注意事项。 文件下载: TPS54372EVM-215.pdf 1. 模块概述 TPS54372EVM - 215评估模块采用了TPS54372跟踪/终止同步
    的头像 发表于 04-24 14:55 120次阅读

    单片机常用的14C语言算法分享

    ) 基本思想:(将相邻两个数比较,小的调到前头) 1)有n个数(存放在数组a(n)),第一趟将每相邻两个数比较,小的调到前头,经n-1次两两相邻比较后,最大的数已“沉底”,放在最后一
    发表于 01-29 06:59

    C语言插入排序算法和代码

    和待插入的元素。第一轮时,将第一元素作为排序好的子数组,插入第二元素;第二轮,将前两
    发表于 01-15 06:44

    HD3SS215:高速差分开关的卓越之选

    HD3SS215:高速差分开关的卓越之选 在当今高速数据传输的时代,一款性能出色的差分开关对于确保信号完整性和系统稳定性至关重要。今天,我们就来深入了解一下TI推出的HD3SS215高速差分开
    的头像 发表于 01-14 09:40 616次阅读

    HD3SS215:高速差分开关的卓越之选

    和灵活的配置,成为了众多应用场景的理想选择。本文将深入介绍HD3SS215的特点、应用、详细规格以及设计要点,为电子工程师们在实际设计中提供有价值的参考。 文件下载: hd3ss215.pdf 一、HD3SS
    的头像 发表于 12-22 14:30 1029次阅读

    TUSB215-Q1 USB 2.0高速信号调节器:技术解析与应用指南

    传输通道的ISI信号损失,助力通过USB电气合规性测试。下面,我们就来详细了解这款器件。 文件下载: tusb215-q1.pdf 一、TUSB215-Q1特性亮点 1. 汽车级可靠性
    的头像 发表于 12-18 16:40 425次阅读

    TUSB215:USB 2.0 高速信号调节器的卓越之选

    TUSB215:USB 2.0 高速信号调节器的卓越之选 在当今数字化飞速发展的时代,USB 接口作为数据传输和设备连接的重要桥梁,被广泛应用于各种电子设备。然而,随着传输距离的增加和信号的衰减
    的头像 发表于 12-18 16:40 909次阅读

    线性搜索与二分搜索介绍

    线性搜索(Linear Search):从数组的第一元素开始,依次将当前元素与目标值进行比较,直到找到目标值或搜索完整个数组。 二分搜索(
    发表于 12-01 07:36

    数组的初体验

    名称[1] = 元素2; 数组名称[n-1] = 元素n; 我们将数据放到数组之后又如何获取数组
    发表于 11-25 08:06

    二维数组介绍

    定义是这样的: int array[n][m] 访问: array[a] 那么被访问元素地址的计算方式就是: array + (m * a + b) 这个就是二维数组在内存的本质,其实和一维
    发表于 11-25 07:42

    数组程序无法运行怎么解决?

    主控是103,程序定义一const类型 128k只读数组,放在flash上,程序无法运行,堆栈都初始化不了,在keil编译下正常,在rtthread studio下编译无法运行,求
    发表于 09-15 06:21

    EDS分析元素的基本原理

    EDS对元素定性分析高能电子束入射到样品表面部分区域,这部分区域表层的原子,其壳层(K、L层等)的电子被激发到较高能量的外壳层(L、M层等)或者原子外成为二次电子,该原子处于激发态(高能量),为使
    的头像 发表于 08-08 11:41 1589次阅读
    EDS分析<b class='flag-5'>元素</b>的基本原理

    请问K230和K230D有哪些区别?

    路 ADC K230D 芯片封装是 11x11,K230 芯片封装为 13x13 您可参考资料下载专区的K230芯片引脚定义的 5
    发表于 06-11 06:46

    现在有一三维数组rgb,怎么通过k230把他转变成图片显示出来?

    现在有一三维数组rgb,怎么通过k230把他转变成图片显示出来
    发表于 06-10 08:17

    M215HGE-L31规格书

    电子发烧友网站提供《M215HGE-L31规格书.pdf》资料免费下载
    发表于 06-04 17:27 0次下载