算法:分治(快排)题目练习

目录

题目一:颜色分类

题目二:排序数组

题目三:数组中的第k个最大元素

题目四:库存管理III


题目一:颜色分类

给定一个包含红色、白色和蓝色、共 n 个元素的数组 nums ,原地对它们进行排序,使得相同颜色的元素相邻,并按照红色、白色、蓝色顺序排列。

我们使用整数 0、 1 和 2 分别表示红色、白色和蓝色。

必须在不使用库内置的 sort 函数的情况下解决这个问题。

示例 1:

输入:nums = [2,0,2,1,1,0]
输出:[0,0,1,1,2,2]

示例 2:

输入:nums = [2,0,1]
输出:[0,1,2]

提示:

  • n == nums.length
  • 1 <= n <= 300
  • nums[i] 为 01 或 2

解法:三指针

这道题和前面算法的第一题:移动零那一题思想类似,移动零是将数组分为两部分,缺点是如果遇到重复元素效率就很低了,而这里是将一个数组分为三部分,是三个指针最终将整个数组划分为满足题意的3部分,完美解决了出现重复元素的情况(i 直接++即可):

首先定义三个指针,分别是:left、right、i
用 i 来扫描整个区域,left 下标所指向的元素是0这个区域的最右侧,right 是2这个区域的最左侧

当 i 遍历结束时,left和right指针停的位置,就可以将数组分为三部分,但是在遍历过程中,三个指针可以将整个数组分为以下4部分,由left和right代表的含义就可以很清楚的划分:

[0, left]:全为0
[left+1, i-1]:全为1
[i, right-1]:待扫描
[right, n-1]:全为2

所以根据上述的4个区域,将下面讨论 i 遍历数组时可能出现的情况:

nums[i] == 0:swap[++left, i++]
nums[i] == 1:i++;
nums[i] == 2:swap[--right, i]

nums[i] == 0时,先++left,再交换 i 与 left 指向的元素,再i++,优化为swap[++left, i++]
nums[i] == 1时,不用做其他操作,直接i++
nums[i] == 2时,right先--,再与 i 交换,此时 i 指向的元素是right从右边交换过来的,是未扫描的元素,所以 i 不需要++,继续循环判断即可

并且整个循环结束的条件是 i < right,而不是 i < n,因为 right 表示的是2这个区域的最左侧,所以当 i 遇到 right 时,就表示已经遍历完这个数组了 

left,right,i初始位置如下图所示:

代码如下:

class Solution {
public:
    void sortColors(vector<int>& nums) {
        int n = nums.size();
        //left初始值为-1,right初始值为n
        int left = -1, right = n, i = 0;
        while(i < right)
        {
            if(nums[i] == 0) swap(nums[++left], nums[i++]);
            else if(nums[i] == 1) i++;
            else swap(nums[--right], nums[i]);
        }
    }
};

题目二:排序数组

给你一个整数数组 nums,请你将该数组升序排列。

示例 1:

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

示例 2:

输入:nums = [5,1,1,2,0,0]
输出:[0,0,1,1,2,5]

解法:快排(数组分三块的思想)

之前学习的快排是找一个基准值key,将数组分为2部分,再在其中一部分再找一个基准值key1,继续分为2部分,以此类推,如下所示:

这种方式如果在数组全是重复元素的情况下,就会退化成O(N^2),因为每次都取的最右侧的元素

这道题采用数组分三块的思想,实现快排:

这种方式能够解决出现重复数据时效率很低的问题,因为如果都是重复数据,key的取值就是该元素,排序完一次后,数组中都是=key的区域,而这种方式中我们需要排的是 <key 和 >key 的区域,但是这种情况下没有这两个区域,所以排序结束,仅仅排序了一次,所以如果都是重复数据的时间复杂度是O(N)

分为三部分,左边全是小于key,右边全是大于key,剩余的中间区域就不需要管了,因为左边和右边都划分好了,中间也就划分好了

同样定义三个指针,left、right、i

i来扫描这个数组,left表示小于key的最左侧,right表示大于key的最右侧

所以在扫描数组时分为三步:

nums[i] < key:swap[++left, i++]
nums[i] == key:i++
nums[i] > key:swap[--right, i]

这三步与上一题一模一样,就不细说了

此题还有一个步骤,就是选择key值,之前学过取最左侧的数、取最右侧的数、三数取中等方式,这里采用优化的方式:用随机的方式选择基准的元素

先使用srand种一个随机数种子,再随机得到一个随机数r,使用r%(right - left + 1) + left,得到一个随机数,r就是我们所找的基准值key

代码如下:

class Solution 
{
public:
    vector<int> sortArray(vector<int>& nums) 
    {
        srand(time(nullptr));//生成随机数种子
        qsort(nums, 0, nums.size()-1);
        return nums;
    }
    //数组分三块思想的快排
    void qsort(vector<int>& nums, int l, int r)
    {
        if(l >= r) return;

        int n = nums.size();
        int left = l - 1, right = r + 1, i = l;//数组分三块
        int key = getRandom(nums, l ,r);
        while(i < right)
        {
            if(nums[i] < key) swap(nums[++left], nums[i++]);
            else if(nums[i] == key) i++;
            else swap(nums[--right], nums[i]); 
        }
        //此时分为了[l, left] [left+1, right-1] [right, r]三部分
        //只需要继续划分[l, left]和[right, r]这两部分即可,因为中间部分就是==key的
        qsort(nums, l, left);
        qsort(nums, right, r);
    }
    //用随机的方式选择基准的元素
    int getRandom(vector<int>& nums, int left, int right)
    {
        int r = rand(); //得到一个随机数r
        return nums[r % (right - left + 1) + left];
    }
};

题目三:数组中的第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

求数组中的第 k 哥最大元素,也就是俗称的topK问题

topK问题有四类,分别是:第k大、第k小、前k大、前k小,要解决topK问题,一般有两种方法,堆排序(O(N*logN))或是基于快排的快速选择算法(O(N))

如果规定了必须使用时间复杂度为O(N)的算法,那就只能使用快排,否则也可以使用堆排序解决

下面具体说说快排是怎么解决这个题目的:

优化的快排将数组分为3部分,基准元素是key,三部分分别是 < key,== key,> key,由于求的是第k大的元素,那么每次判断只需要判定这个元素会落到哪一部分,就能够排除其他两部分,从而效率非常高

假设 < key,== key,> key 这三部分分别有a、b、c个元素,所以下面根据元素个数分情况讨论,从右侧区域开始判断,因为右侧区域是大元素的集合

①:c >= k,说明第k大就在这个 > key 的区域里,此时取[right, r]区域中找第 k 大的元素即可
②:b + c >= k,说明第k大的元素在== key的区域中,此时就不需要比较了,直接返回key即可,因为这个区域的数大小都是key
③:走到这里,说明①②都不成立,所以需要去[l, left]区域找
第 k - b -c 大的元素

此题的解决方式就是在上一题的快排的基础上实现的

代码如下:

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

    int qsort(vector<int>& nums, int l, int r, int k)
    {
        if(l == r) return nums[l];
        // 随机选择基准元素
        int key = getRandom(nums, l, r);
        // 根据基准元素将数组分为3块
        int left = l - 1, right = r + 1, i = l;
        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, b = right - left - 1;
        if(c >= k) return qsort(nums, right, r, k);
        else if((b + c) >= k) return key;
        else return qsort(nums, l, left, k - b - c);//注意不是k,而是k-b-c
    }

    int getRandom(vector<int>& nums, int left, int right)
    {
        int r = rand();
        return nums[r % (right - left + 1) + left];
    }
};

题目四:库存管理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]

这道题,观察给出的题目信息,其实也是一个topK问题,只不过这里的topK问题是求前k个最小的数

此题有很多解法,例如:

解法一:排序,最后取出前k个最小的数,时间复杂度O(NlogN)

解法二:堆排序,时间复杂度O(Nlogk)

解法三:快速选择算法,时间复杂度O(N)

这里只实现快速选择算法,前两种都比较简单

依然是随机选择基准元素 + 把数组分三块的思想,依旧是分为三部分,分别是 < key,== key,> key,这三部分分别有a、b、c个元素,并且left指向的是最左侧区域的最后一个值,right表示最右侧区域的第一个值,如下所示:

因为此题求的是前k小的元素,所以先考虑 < key 的这个区域,步骤如下:

①:a > k,说明就在< key 的这个区域,在[l, left]区域中查找
②:a + b >= k,直接返回
③:走到这说明①②都不满足,所以在 >key 这个区域即[right, r]中,找k - a - b 个最小元素即可

代码如下:

class Solution 
{
public:
    vector<int> inventoryManagement(vector<int>& stock, int cnt) 
    {
        srand(time(nullptr));
        qsort(stock, 0, stock.size()-1, cnt);
        //最后将前k个元素返回即可
        return {stock.begin(), stock.begin() + cnt};
    }

    void qsort(vector<int>& nums, int l, int r, int k)
    {
        if(l >= r) return;
        //随机选择一个基准元素
        int key = getRandom(nums, l, r);
        //数组分三块
        int left = l - 1, right = r + 1, i = l;
        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, b = right - left - 1;
        if(a > k) qsort(nums, l, left, k);
        else if(a + b >= k) return;
        else qsort(nums, right, r, k - a - b);
    }

    int getRandom(vector<int>& nums, int left, int right)
    {
        return nums[rand() % (right - left + 1) + left];
    }
};

分治中,关于快排的题目到此结束


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/712006.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

红队攻防渗透技术实战流程:中间件安全:JettyJenkinsWeblogicWPS

红队攻防渗透实战 1. 中间件安全1.1 中间件-Jetty-CVE&信息泄漏1.2 中间件-Jenkins-CVE&RCE执行1.2.1 cve_2017_1000353 JDK-1.8.0_291 其他版本失效1.2.2 CVE-2018-10008611.2.3 cve_2019_100300 需要用户帐号密码1.3 中间件-Weblogic-CVE&反序列化&RCE1.4 应…

驱动开发(四):Linux内核中断

驱动开发系列文章&#xff1a; 驱动开发&#xff08;一&#xff09;&#xff1a;驱动代码的基本框架 驱动开发&#xff08;二&#xff09;&#xff1a;创建字符设备驱动 驱动开发&#xff08;三&#xff09;&#xff1a;内核层控制硬件层 驱动开发&#xff08;四&#xf…

2024FIC决赛

容器密码&#xff1a;2024Fic~Competition~Finals杭州&Powered~By~HL! 案件背景: 2023年3月15日凌晨,受害人短视频平台上看到一段近期火爆的交通事故视频&#xff0c;留言后有人通过私信联系&#xff0c;称有一个赚大钱的机会&#xff0c;该人自称李某&#xff0c;提议让…

如何通过抖音自动评论精准获客实现业务增长?这些方法值得一试!

在当今竞争激烈的商业环境中&#xff0c;企业若想脱颖而出&#xff0c;就必须掌握精准获客的艺术。精准获客&#xff0c;即通过精确的市场定位和营销策略&#xff0c;吸引并保留最有可能成为客户的目标群体。它不仅能提高转化率&#xff0c;还能有效降低营销成本&#xff0c;是…

实况:老菜鸟自力更生从零开始重学spring目标是画出一张唬人大图(二、源码下载编译)

前情提要&#xff1a;调试前的基础知识梳理 速览 “Spring”包含哪些东西源码下载源码编译1、编译工具选择&#xff1a;gradle2、使用gradle编译spring并导入idea预编译spring-oxm导入IDEA确认合适的jdk版本排除spring-aspects模块 开始调试 “Spring”包含哪些东西 可以明确的…

LVS负载均衡:理解IPVS和IPVSADM的内部工作原理

LVS 负载均衡工作模式 LVS&#xff08;Linux Virtual Server&#xff09; 共有三种工作模式&#xff1a;DR、Tunnel、NAT。 DR&#xff08;Direct Routing&#xff09;&#xff1a; 技术原理&#xff1a;DR模式下&#xff0c;LVS调度器接收到请求后&#xff0c;直接通过MAC地址…

Kali中安装和使用docker的学习笔记

一、常见命令 ctrl 、shift、 &#xff1a; 窗口变大&#xff1b; ctrl 、- &#xff1a;窗口变小&#xff1b; ctrl L&#xff1a; 清屏 &#xff1b; sudo su : 切换root 用户&#xff1b; ip addr / ifconfig: 获取IP地址&#xff1b; systemctl start ssh…

探索Python的多媒体解决方案:ffmpy库

文章目录 探索Python的多媒体解决方案&#xff1a;ffmpy库一、背景&#xff1a;数字化时代的多媒体处理二、ffmpy&#xff1a;Python与ffmpeg的桥梁三、安装ffmpy&#xff1a;轻松几步四、ffmpy的五项基本功能1. 转换视频格式2. 调整视频质量3. 音频转换4. 视频截图5. 视频合并…

Mybatis框架中结果映射resultMap标签方法属性收录

Mybatis框架中结果映射resultMap标签收录 在MyBatis框架中&#xff0c;resultMap 是一种强大的机制&#xff0c;用于将数据库结果集映射到Java对象上。它允许你定义如何将查询结果中的列映射到Java对象的属性上&#xff0c;尤其是当数据库表的字段名与Java对象的属性名不一致时…

【太原理工大学】软件系统安全—分析题

OK了&#xff0c;又是毫无准备的一场仗&#xff0c;我真是ありがとうございます 凸^o^凸 根据前几年传下来的信息&#xff0c;所谓“分析”&#xff0c;就是让你根据情节自行设计&#xff0c;例如如何设计表单等&#xff0c;这类多从实验中出&#xff0c;王老师强调好好做实验一…

自然抽样和平顶抽样

自然抽样和平顶抽样是两种信号处理和采样技术&#xff0c;它们在音频信号处理、信号重建以及数字信号处理中有着不同的应用。 1. 自然抽样&#xff08;也称为理想抽样或无失真抽样&#xff09;&#xff1a;样值脉冲的幅度随原始信号m(t)的幅度而变&#xff1b; 自然抽样过程的…

个人网站制作 Part 26 添加在线日历功能 | Web开发项目添加页面缓存

文章目录 &#x1f469;‍&#x1f4bb; 基础Web开发练手项目系列&#xff1a;个人网站制作&#x1f680; 添加在线日历功能&#x1f528;使用日历服务&#x1f527;步骤 1: 选择日历服务&#x1f527;步骤 2: 安装FullCalendar&#x1f527;步骤 3: 创建FullCalendar组件&…

【2024最新华为OD-C/D卷试题汇总】[支持在线评测] 生成哈夫曼树(100分) - 三语言AC题解(Python/Java/Cpp)

🍭 大家好这里是清隆学长 ,一枚热爱算法的程序员 ✨ 本系列打算持续跟新华为OD-C/D卷的三语言AC题解 💻 ACM银牌🥈| 多次AK大厂笔试 | 编程一对一辅导 👏 感谢大家的订阅➕ 和 喜欢💗 📎在线评测链接 生成哈夫曼树(100分) 🌍 评测功能需要订阅专栏后私信联系清…

数据中心布线管理:预标记线缆与移动扫描技术的融合

随着信息技术的飞速发展&#xff0c;数据中心布线管理面临着前所未有的挑战。传统的布线管理方式已无法满足现代数据中心高效、准确和可靠的需求。在这样一个背景下&#xff0c;预标记线缆与移动扫描技术的结合&#xff0c;为数据中心布线管理带来了革命性的解决方案。 布线管理…

基于System-Verilog点亮LED灯

文章目录 一、System-Verilog介绍1.1System-Verilog 二、简单的语法介绍2.1接口实例2.2全局声明和语句实例2.3时间单位和精度2.4用户定义的类型2.5 枚举类型 三、流水灯参考 一、System-Verilog介绍 1.1System-Verilog SystemVerilog是一种硬件描述和验证语言&#xff08;HDV…

stm32f103 HAL库 HC-SR04测距

目录 一、实现测距二、添加TIM3控制LED根据距离以不同频率闪烁三、观察时序Modebus协议12路超声波雷达设计方案1. 系统架构设计2. 硬件设计3. 软件设计4. 通信协议设计5. 用户接口6. 安全和冗余7. 测试和验证8. 电源和物理封装9. 文档和支持 一、实现测距 配置时钟 配置定时器…

八、BGP

目录 一、为何需要BGP&#xff1f; 二、BGP 2.1、BGP邻居 2.2、BGP报文 2.3、BGP路由 2.4、BGP通告遵循原则 2.5、BGP实验 第一步&#xff1a;建立邻居 第二步&#xff1a;引入路由 BGP路由黑洞 路由黑洞解决方案 1、IBGP全互联 2、路由引入 3、MPLS 多协…

【Python】数据处理:Matplotlib绘图

Matplotlib是Python强大的数据可视化工具库&#xff0c;类似于MATLAB语言。Mat-lotlib提供了一整套与MATLAB相似的命令API&#xff0c;十分适合进行交互式制图&#xff0c;而且也可以方便地将它作为绘图控件&#xff0c;嵌入GUI应用程序中。 Matplotlib是神经生物学家John D.Hu…

轻松拿捏C语言——【关机代码】

&#x1f970;欢迎关注 轻松拿捏C语言系列&#xff0c;来和 小哇 一起进步&#xff01;✊ &#x1f389;创作不易&#xff0c;请多多支持&#x1f389; &#x1f308;感谢大家的阅读、点赞、收藏和关注&#x1f495; &#x1f339;如有问题&#xff0c;欢迎指正 我们可以通过写…

c++编程(16)——STL(4)deque

欢迎来到博主的专栏&#xff1a;c编程 博主ID&#xff1a;代码小豪 文章目录 dequedeque的优劣势deque的操作constructor元素访问 deque deque的全称是double ended queue&#xff0c;译为双端队列&#xff0c;如何理解这个双端呢&#xff1f;我们以vector为例&#xff0c;vec…