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

[Lc7_分治-快排] 快速选择排序 | 数组中的第K个最大元素 | 库存管理 III

目录

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

题解

代码

2.库存管理 III

代码


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

题目链接:215. 数组中的第K个最大元素

题目分析:

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

请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。

你必须设计并实现时间复杂度为 O(n) 的算法解决此问题。

示例 1:

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

示例 2:

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

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

其实就是一个TopK问题。

TopK问题四种问法:

  • 第 k 个最大的元素✔️
  • 第 k 个最小的元素
  • 前 k 个最大的元素
  • 前 k 个最小的元素✔️

可以使用堆排序, 时间复杂度O(nlogn)

还有一种就是快速选择算法,快速选择算法是基于快排实现的。时间复杂度O(n)。

题解

一定要 把数组分三块+随机选择基准元素 掌握,才能懂这道题。

  • 核心操作还是选择一个基准元素key,将数组分成< key , = key ,> key 三块区域。
  • 在这道题中我们是要找到第K大元素,这个第K大元素有可能落在三个区域中的任何一个区域。
  • 如果有一种方法能够 确定第K大元素能够落在那个区域,那另外两个区域就不用考虑了。
  • 仅需在确定的区域里面递归找。所以说它是比快排更快的一种算法。

接下来重点就是 如何确定第K大元素落在左边区域、中间区域、还是右边区域。

此时我们 先统计出每个区域中元素的个数,假设左边区域元素个数为 a,中间区域元素个数为 b,右边区域元素个数为 c。

此时就分三种情况讨论:

  • 因为求第K大,所以可以 先考虑右边区域
  • 因为右边区域都是大元素聚集地,第K大最有可能在右边区域。

第一种情况:

  • 如果第K大是落在右边区域,此时 c >= K(例如 c 为前 5 大,K 为 找出第三大的)
  • 因为c表示大元素有多少个,而K表示找第K大的元素。如果 c >= K ,那第K大一定是落在右边区域。
  • 此时我们仅需到[right,r],找第K大。

第二种情况:

  • 如果到了第二情况那第一种情况一定是不成立的。
  • 如果第K大是落在中间区域,此时 b + c >= K,又因为第一种情况不满足,所有第K大一定是落在中间区域。
  • 此时就找到了也不用递归了。直接返回就可以。

第三种情况:

  • 第一、第二种情况全部不成立,第K大一定落在左边区域,去[l,left]找
  • 但是此时并不是去找第K大了,本来是想在整个区间找第K大,但是第K大元素确定不在中间区域和右边区域,中间和右边区域都要被抛弃
  • 此时去找左边区间找的是第 K - b - c 大的元素

代码

class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) 
    {
        int n=nums.size();
        srand((unsigned int)time(nullptr));
        return QuickSort(nums,0,n-1,k);
    }

    int QuickSort(vector<int>& nums,int l,int r,int k)
    {
        if(l==r) return nums[l];
        //!!!!!!!递归的 终止情况

        int key=nums[rand()%(r-l+1)+l];

        int i=l,left=l-1,right=r+1;//初始化在 区间边界之外

        while(i<right)
        {
            if(nums[i]<key)
                swap(nums[++left],nums[i++]);
            else if(nums[i]==key)
                i++;
            else
                swap(nums[--right],nums[i]);
        }
        int c=r-right+1;
        int b=right-1-left;

        if(c>=k)
            return QuickSort(nums,right,r,k);
        else if(b+c>=k && c<k)
            return key;
        else 
            return QuickSort(nums,l,left,(k-b-c));//return调用递归
    }

};

2.库存管理 III

题目链接:LCR 159. 库存管理 III

题目分析:

仓库管理员以数组 stock 形式记录商品库存表,其中 stock[i] 表示对应商品库存余量。请返回库存余量最少的 cnt 个商品余量,返回 顺序不限

示例 1:

输入:stock = [2,5,7,4], cnt = 1
输出:[2]

示例 2:

输入:stock = [0,2,3,6], cnt = 2
输出:[0,2] 或 [2,0]

即 前面说到的返回 前 k 小的元素


实际上这也是一个TopK问题,让找前K个最小元素。注意返回结果并不考虑顺序问题。

算法原理:

  • 解法一:排序 ,时间复杂度O(nlogn)
  • 解法二:堆 ,时间复杂度O(nlogk)
  • 解法三:快速选择算法,时间复杂度O(n)

数组分三块+随机选择基准元素。

选择一个基准元素key,将数组分成< key , = key ,> key 三块区域。找前K个最小的元素,落在三个区域中任何一个。

统计除每个区域元素个数,然后选择去那个区域找。

分三种情况:

  • 因为前K下最小元素 最有可能出现在左边区域,因此先判断左边区域
  • a >= K ,去[l,left] 找前K个最小元素
  • b + a >= K ,直接返回
  • 1、2都不成立,去[right,r] 找 K - a - b 个最小元素

代码

class Solution {
public:
    vector<int> inventoryManagement(vector<int>& stock, int cnt) 
    {
        int n=stock.size();
        srand((unsigned int)time(nullptr));
        if(cnt==0) return {};
        return QuickSort(stock,0,n-1,cnt);
    }

    vector<int> QuickSort(vector<int>& nums,int l,int r,int k)
    {
//1.
        if(l >= r) return {nums[l]}; // 基础case返回单元素

        int key=nums[rand()%(r-l+1)+l];
        //快排
        int i=l,left=l-1,right=r+1;


        while(i<right)
        {
            if(nums[i]<key)
                swap(nums[++left],nums[i++]);
            else if(nums[i]==key)
                i++;
            else    
                swap(nums[--right],nums[i]);
        }

        int a=left-l+1;
        int b=right-1-left;

        if(a>=k)
            return QuickSort(nums,l,left,k);
         else if(a + b >= k) 
            return vector<int>(nums.begin()+l, nums.begin()+l+k); 
            // 直接取前k个

         else 
         {
            vector<int> res(nums.begin()+l, nums.begin()+right); 
            // 全取左&等于段
            vector<int> rightRes = QuickSort(nums, right, r, k - a - b);
            res.insert(res.end(), rightRes.begin(), rightRes.end());
            return res;
        }
    }
};

返回 前 k 个元素,在返回 第 K 个元素基础上,做的嵌套改动

相关文章:

  • p5.js:模拟 n个彩色小球在一个3D大球体内部弹跳
  • MySQL如何给其他账号分配权限?
  • Java基础系列:深入理解八大基本数据类型及避坑指南
  • python 程序一次启动有两个进程的问题(flask)
  • Jetpack Compose — 入门实践
  • 完全日期(日期枚举问题)--- 数学性质题型
  • Linux系统重置密码
  • Kubernetes中的微服务
  • 报表DSL优化,享元模式优化过程,优化效果怎么样?
  • SpringCloud——Consul服务注册与发现
  • powershell@宝塔面板批量建站脚本@批量设置@批量部署伪静态设置
  • 【AD】5-12 Object元素的隐藏与显示
  • c语言操作符
  • 前端项目中export和import的作用
  • 网安知识点
  • 从Manus到OpenManus:多智能体协作框架如何重构AI生产力?
  • 【Linux内核系列】:深入解析输出以及输入重定向
  • 分布式锁—7.Curator的分布式锁一
  • Java为什么是跨平台的
  • 《深度剖析架构蒸馏与逻辑蒸馏:探寻知识迁移的差异化路径》
  • 惠州营销网站建设公司/媒体发稿公司
  • 网站文章要求/写软文能赚钱吗
  • ssh框架可以做网站么/微商引流人脉推广软件
  • 空间做网站/驻马店网站seo
  • 公司网站开发建设什么会计科目/网络营销软件代理
  • 宁波电商平台网站建设/定制营销型网站建设