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

目录

题目一:颜色分类

题目二:排序数组

题目三:数组中的第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初始值为nint 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(); //得到一个随机数rreturn 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];}
};

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


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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【回文 马拉车】214. 最短回文串

本文涉及知识点 回文 马拉车 LeetCode214. 最短回文串 给定一个字符串 s&#xff0c;你可以通过在字符串前面添加字符将其转换为回文串。找到并返回可以用这种方式转换的最短回文串。 示例 1&#xff1a; 输入&#xff1a;s “aacecaaa” 输出&#xff1a;“aaacecaaa” 示…

【单元测试】Spring Boot 的测试库

Spring Boot 的测试库 1.了解回归测试框架 JUnit2.了解 assertThat3.了解 Mockito4.了解 JSONPath5.测试的回滚 单元测试&#xff08;unit test&#xff09;是为了检验程序的正确性。一个单元可能是单个 程序、类、对象、方法 等&#xff0c;它是应用程序的最小可测试部件。 单…

[大模型]XVERSE-7B-chat Transformers 推理

XVERSE-7B-Chat为XVERSE-7B模型对齐后的版本。 XVERSE-7B 是由深圳元象科技自主研发的支持多语言的大语言模型&#xff08;Large Language Model&#xff09;&#xff0c;参数规模为 70 亿&#xff0c;主要特点如下&#xff1a; 模型结构&#xff1a;XVERSE-7B 使用主流 Deco…

用于每个平台的最佳WordPress LMS主题

你已选择在 WordPress 上构建学习管理系统 (LMS)了。恭喜&#xff01; 你甚至可能已经选择了要使用的 LMS 插件&#xff0c;这已经是成功的一半了。 现在是时候弄清楚哪个 WordPress LMS 主题要与你的插件配对。 我将解释 LMS 主题和插件之间的区别&#xff0c;以便你了解要…

如何打开pak文件-翻译pak语言包

最近碰到一些程序的语言包是pak格式&#xff0c;用Notepad打开全是乱码&#xff0c;百度搜索了一下&#xff0c;pak是一种少见的压缩文件格式&#xff0c;是pak Quake系列游戏所采用的一种特殊压缩包格式&#xff0c;由Quake游戏公司开发&#xff0c;用高版本的winrar可以打开&…

测试 halcon算子 derivate_gauss 高斯一阶导数卷积

参上了 matlab fileexchange 有人上传了高斯 dx,dy一阶导卷积代码 卷积核的计算我修改成了核元素绝对值求做分母 归一化 和halcon的 derivate_gauss算子的计算结果对别如下 还是不知道怎么做到两者结果一致. 测试图像: 我的: halcon的: 获取两份图像的灰度值到数组并做对应位…

即时聊天系统

功能描述 该项目是一个前后端分离的即时聊天项目&#xff0c;前端采用vue2、后端使用springboot以mysql8.0作为数据库。 项目功能包含了单聊、群聊功能。在此基础上增加了对好友的功能操作&#xff0c;如备注设为通知、视频聊天、语音聊天、置顶、拉入黑名单、清空聊天记录等。…

【面试干货】Integer 和 int 的区别

【面试干货】Integer 和 int 的区别 1、基本类型与包装类型2、内存占用3、自动装箱与拆箱4、null 值5、常量池6、总结 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; 在Java中&#xff0c;Integer 和 int 是两种不同类型的变量&#xff0c;…

leetcode LRU 缓存

leetcode: LRU 缓存 LRU 全称为 Least Recently Used&#xff0c;最近最少使用&#xff0c;常常用于缓存机制&#xff0c;比如 cpu 的 cache 缓存&#xff0c;使用了 LRU 算法。LRU 用于缓存机制时&#xff0c;关键的是当缓存满的时候有新数据需要加载到缓存的&#xff0c;这个…

自动化测试断言

自动化判断测试用例的执行的结果是否成功&#xff0c;是通过判断测试得到的实际结果与预期结果是否相等决定的。这个时候就用到了断言。 检查点分为两个&#xff0c;一个是页面级别的检查&#xff0c;包括网页的标题和网址&#xff0c;以及是否包含某个文字 另一个检查点是页…

CSS从入门到精通——动画:CSS3动画延迟和完成后状态的保持

目录 任务描述 相关知识 动画状态 动画完成时的状态 动画延迟 编程要求 任务描述 本关任务&#xff1a;用 CSS3 实现小车等待红绿灯的效果。效果图如下&#xff1a; 相关知识 为了完成本关任务&#xff0c;你需要掌握&#xff1a;1.动画状态&#xff0c;2.动画完成时的状…

[Cloud Networking] SPDY 协议

文章目录 1. 背景2. SPDY 之前3. SPDY 项目目标4. SPDY 功能特点4.1 SPDY基本功能4.2 SPDY高级功能 1. 背景 TCP是通用的、可靠的传输协议&#xff0c;提供保证交付、重复抑制、按顺序交付、流量控制、拥塞避免和其他传输特性。 HTTP是提供基本请求/响应语义的应用层协议。 不…

Linux下的串口通信

串口通信 基础知识&#xff1a; 什么是串口&#xff1f; 串口全称串行通信接口&#xff0c;是一种常用于电子设备之间通信的异步&#xff0c;全双工接口&#xff0c;典型的串口通信只需要 3 根线&#xff0c;分别是地线 (GND)&#xff0c;发送线(TX)&#xff0c;接收线(RX)。如…

【react小项目】bmi-calculator

bmi-calculator 目录 bmi-calculator初始化项目01大致布局01代码 02完善样式02代码 03输入信息模块03代码 04 使用图表04代码 05详细记录信息渲染05代码 06 让数据变成响应式的06-1输入框的数据处理06-2图表&#xff0c;和记录信息的区域数据处理 07 删除功能&#xff0c;撤销功…

基于C#开发web网页管理系统模板流程-主界面统计功能完善

点击返回目录-> 基于C#开发web网页管理系统模板流程-总集篇-CSDN博客 前言 紧接上篇->基于C#开发web网页管理系统模板流程-主界面管理员入库和出库功能完善_c#web程序设计-CSDN博客 统计功能是管理系统很常见的功能&#xff0c;例如仓库管理系统要统计某时间段的出入库以…

QT信号与槽/窗口组件优化/使用QT制作QQ登录界面

使用手动连接&#xff0c;将登录框中的取消按钮使用第二中连接方式&#xff0c;右击转到槽&#xff0c;在该槽函数中&#xff0c;调用关闭函数 将登录按钮使用qt4版本的连接到自定义的槽函数中&#xff0c;在槽函数中判断u界面上输入的账号是否为"admin"&#xff0c;…

永磁同步直线电机(PMLSM)控制与仿真3-永磁同步直线电机数学三环控制整定

文章目录 1、电流环参数整定2、速度环参数整定3、位置环参数整定 写在前面&#xff1a;原本为一篇文章写完了永磁同步直线电机数学模型介绍&#xff0c;永磁同步直线电机数学模型搭建&#xff0c;以及永磁同步直线电机三环参数整定及三环仿真模型搭建&#xff0c;但因为篇幅较长…

HTML前端

html 超文本标记语言 文本&#xff1a;文字字符 超文本&#xff1a;网页内容 标记&#xff1a;标签 标识 提供许多标签&#xff0c;不同标签功能不同&#xff0c;网页就是通过这些标签描述出来的&#xff0c;最终由浏览器解释运行我们看到的网页 <!-- html注释<!DO…

C++ 50 之 继承中的对象模型

继承中的对象模型 在C编译器的内部可以理解为结构体&#xff0c;子类是由父类成员叠加子类新成员而成&#xff1a; #include <iostream> #include <string> using namespace std;class Base03{ public:int m_a; protected:int m_b; private:int m_c; // 哪怕是…

lua对接GPT4实现对话

演示效果&#xff1a; 准备材料&#xff1a; 1、FastWeb网站开发服务&#xff1a;fwlua.com 2、一台服务器 该示例使用开源项目&#xff1a;fastweb 实现。 代码比较简单&#xff0c;主要是两部分&#xff0c;一个lua代码和一个html页面&#xff0c;用来用户发起请求和后台…