IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> leetcode_215 -> 正文阅读

[数据结构与算法]leetcode_215

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

题目阐述:
给定整数数组nums和整数k,请返回数组中第k个最大的数组。
请注意,你需要找的是数组排序后的第k个最大的元素,而不是第k个不同的元素

Eg1:

input: [3,2,1,5,6,4] 和 k = 2

output: 5

Eg2:

input: [3,2,3,1,2,4,5,5,6] 和 k = 4

output: 4

notes:

  • 1 <= k <= nums.length <= 104
  • -104 <= nums[i] <= 104

解题思路

这道题刚拿到手,看到难度竟然是一道中等题目
开始想:既然我已经学过了C++ stl库,这道题真的是太简单了,直接sort呗。

但为了给这道题一点面子。。这里会用到三种方法来解决这个看似非常简单的问题

解题是主要的,但是我们也不能忽视了效率问题,所以最后会好好地总结一下这三个方法的优良差异

1. 直接暴力排序

就像刚刚说的,直接用stl库里面的sort(快排),对整个数组进行一个排序。

因为sort中,默认为排升序

所以最后返回的应该是下标为 (数组长度 - k)

代码如下:

class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) {
        sort(nums.begin(),nums.end());
        return nums[nums.size() - k];
    }
};

2. 建堆

有一个同样在stl库里常用的容器————priority_queue(优先级队列)
它会在每次我们插入push数据,删除pop数据,自动执行AdjustUpAdjustDown进行建堆。

思路:既然我们要找的是第k个最大的元素
那么我们就构建一个大堆,然后构造k-1次堆就行了
(什么意思呢,我们知道,大堆的每次构建,最上面的top返回值都是当前最大的数字,要找第k个,那么我们只需要构建k-1次堆即可)

想清楚了这个思路,这道题就可以解了XD

class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) {
        priority_queue<int> q(nums.begin(),nums.end());
        for(int i = 0;i < k - 1;i++)
        {
            q.pop();
        }
        return q.top();
    }
};

3. 建堆(优化)

这里其实严格意义来说不能说算是优化,只是说对待不同的情况,这两种方法都得去考虑到

之前学堆的时候看到过一个问题,关于如何海量信息处理

这道题其实还好,题目给的例子都非常的温柔。

但如果让你从10000个数据里面找寻第100个最大的元素
这可够要命的

要知道建堆一次的时间复杂度为O(N)
数量大起来就成了O(N^N),这样问题就很大了

这样就衍生出第三种方法:

既然是海量数据,考虑到我们的问题就是建上面的时间复杂度问题,那么我们就想办法少建几次呗.
我们现在的问题可以转化成这样:
找最大的前k个数

思路:我们先建立一个k个元素大小组成的小堆,然后其余数字x每每和堆头top作比较,大的话就将堆pop掉,然后将x压入堆内。

具体代码如下:

class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) {
        priority_queue<int,vector<int>,greater<int>> q(nums.begin(),nums.begin()+k);
        for(size_t i = k;i < nums.size();i++)
        {
            if(nums[i] > q.top())
            {
                q.pop();
                q.push(nums[i]);
            }
        }
        return q.top();
    }
};

复杂度分析:

1.

第一个办法太简单粗暴了
直接一次快排,时间复杂度为O(N*logN)

2.

第二个方法要计算时间复杂度的话,我们得先知道建堆向下调整算法的各自时间复杂度

建堆的时间复杂度为: O(N)

向下调整算法为: O(logN)

这里可以直接看我的blog
里面有详细计算解释

那么我们计算一下:
建堆需要O(N),要popK次,那么向下调整算法需要O(K*logN)

加起来就是O(N+K*logN)

第二种方法不仅仅是时间复杂度上面的问题,还同时要考虑空间复杂度。

这里建了一个N个数的堆,因为堆的底层就是用一个vector存起来,所以空间复杂度为O(N)

3.

第三个方法也不难计算
首先时间复杂度也是那两项:
建堆+向下调整
这里的向下调整是O((N-K)*logK)

我们这里的N-K已经是最差的情况了

这样时间复杂度就为O(N+(N-K)*logK)

而空间复杂度为O(K)

分析

这样一来我们就不去考虑第一种的方法了,只看第二和第三种。

NK相差不大的时候,其实这两者在时间上的效率和空间上的复杂度并不是相差特别大

但是相差大的时候,时间复杂度还好,空咋复杂度就拉开很大的差距了,一个只需要建立K个大小的,另外一个却要建立N

这就是第三个方法为啥是最优的!

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-09-13 09:31:11  更:2021-09-13 09:31:15 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/17 14:38:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码