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

news2025/1/23 9:31:12

目录

题目一:颜色分类

题目二:排序数组

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

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


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

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

相关文章

Linux_应用篇(19) V4L2 摄像头应用编程

ALPHA/Mini I.MX6U 开发板配套支持多种不同的摄像头&#xff0c;包括正点原子的 ov5640&#xff08;500W 像素&#xff09;、ov2640&#xff08;200W 像素&#xff09;以及 ov7725&#xff08;不带 FIFO、 30W 像素&#xff09;这三款摄像头&#xff0c;在开发板出厂系统上&…

Jupyter Notebook简介

目录 1.概述 2.诞生背景 3.历史版本 4.安装 5.卸载 6.如何使用 7.菜单和菜单项 8.示例 9.未来展望 10.总结 1.概述 Jupyter Notebook是一种基于Web的交互式计算环境&#xff0c;主要用于数据分析、数据科学、机器学习以及探索性编程等领域。允许用户在单个文档中编写…

3.华为trunk和access接口配置

目的&#xff1a;PC1 连通三层交换机LSW1 LSW1配置 [Huawei]vlan batch 10 [Huawei]interface Vlanif 10 [Huawei-Vlanif10]ip address 10.10.10.10 24 [Huawei]int g0/0/1 [Huawei-GigabitEthernet0/0/1]port link-type trunk [Huawei-GigabitEthernet0/0/1]port trunk allow…

嵌入式操作系统_2.嵌入式操作系统的一般架构

1.嵌入式操作系统的概念 嵌入式操作系统通常由硬件驱动程序、调式代理、操作系统内核、文件系统和可配置组件等功能组成&#xff0c;并为应用软件提供标准的API&#xff08;Application Programming Interface&#xff09;接口服务。 2.一般嵌入式操作系统的体系结构 从嵌入…

C#|Maui|BootstrapBlazor|Bootstrap Blazor 组件库改模板 | Bootstrap Blazor 组件库改布局,该怎么改?

先copy一个项目下来&#xff1a;Bootstrap Blazor 组件库 一套基于 Bootstrap 和 Blazor 的企业级组件库 发现不是很满足我的需求&#xff0c;我要把右下角的admin移动到左边去&#xff0c;该怎么移动&#xff1f; 先改代码 点进去到Layout.razor 文档&#xff0c;改成如下&am…

“专业敏捷教练课程” 8月31-9月1日 · CSP-SM认证周末班【晋升高阶享多重福利】

点击标题阅读&#xff1a; 《数字时代下敏捷教练专业技能CSP-SM框架解析》 为什么“模块化分时段”单元教学&#xff1f; ☆ 有脑科学研究资料揭示: 成人学习者持续3.5小时已经达到极限&#xff0c;新模式教学&#xff0c;给学习者留有一些时间和空间去消化吸收&#xff0c;…

清华停招土木,新增地球科学引热议

早在今年2月26日&#xff0c;多个自媒体平台上有人发布消息称“清华大学停止土木工程等专业招生”&#xff0c;引发广泛关注。 在清华大学的官网可以看到下图的公告。 可以看到&#xff0c;清华大学停招土木工程等专业&#xff0c;新增地球系统科学等专业。这一举措引起全网热…

收银系统源码-连锁店收银系统,支持二次开发

千呼新零售2.0系统是零售行业连锁店一体化收银系统&#xff0c;包括线下收银线上商城连锁店管理ERP管理商品管理供应商管理会员营销等功能为一体&#xff0c;线上线下数据全部打通。 适用于商超、便利店、水果、生鲜、母婴、服装、零食、百货等连锁店使用。 一、收银端 支持Wi…

vim 的 map+noremap

经常在 vim 的配置文件中&#xff0c;看到对于改键的设置。 他们的区别主要有两种 1 用于哪种模式。 2 是否用于递归。

一文快速认识环形光源——CCS光源

机器视觉系统中&#xff0c;光源起着重要作用&#xff0c;不同类型的光源应用也不同&#xff0c;选择合适的光源成像效果非常明显。今天我们一起来看看CCS光源——工业用环形光源LDR2系列。 LDR2系列是标准的环形光源&#xff0c;通过采用柔性基板&#xff0c;可创造任意角度。…

每日一题——Python实现PAT甲级1132 Cut Integer(举一反三+思想解读+逐步优化)五千字好文

一个认为一切根源都是“自己不够强”的INTJ 个人主页&#xff1a;用哲学编程-CSDN博客专栏&#xff1a;每日一题——举一反三Python编程学习Python内置函数 Python-3.12.0文档解读 目录 我的写法 正确性和功能性 时间复杂度 空间复杂度 其他点评 总结 我要更强 优化后…

【每日LeetCode】递归、记忆化搜索

递归、记忆化搜索 【leetcode70 爬楼梯】 class Solution {public int climbStairs(int n) {int[] memo new int[n 1];return dfs(n, memo);}private int dfs(int i, int[] memo){if(i < 1){return 1;}if(memo[i] ! 0){return memo[i];}return memo[i] dfs(i-1,memo) d…

成功秘诀曝光:老阳分享选品师赚钱攻略

当谈论老阳分享的外海拼多多选品师项目时&#xff0c;人们往往对其真实性和可行性存有疑问。这个项目被宣传为一个能够稳定创收的机会&#xff0c;但在决定是否投身其中之前&#xff0c;了解其具体运作和实际效果至关重要。 老阳分享作为电商培训平台&#xff0c;旨在培养和支持…

充电学习——0、电源管理

一、设备电源管理&#xff1a; 两种类型 1、系统睡眠模型&#xff1a; 设备驱动作为系统一部分&#xff0c;会跟随系统进入低功耗状态&#xff0c;suspend &#xff08;suspend-to-ram&#xff09; 一些驱动程序可以管理硬件的唤醒事件&#xff0c; 这一特性通过/sys/device/…

GaussDB技术解读——GaussDB架构介绍(四)

目录 11 GaussDB云原生架构 11.1 云原生关键技术架构 11.2 关键技术方案 11.2.1 通信组件 11.2.2 集群管理组件 11.2.3 多租组件 GaussDB架构介绍&#xff08;三&#xff09;从智能关键技术方案、驱动接口关键技术方案等方面对GaussDB架构进行了解读&#xff0c;本篇将…

【CTF Web】CTFShow 探针泄露 Writeup(PHP+探针泄露+信息收集)

探针泄露 10 对于测试用的探针&#xff0c;使用完毕后要及时删除&#xff0c;可能会造成信息泄露 解法 查看网页源代码。 view-source:https://11170dfe-84c7-4fde-b1ca-5d1ec3dd7570.challenge.ctf.show/没有找到有用的信息。 用 dirsearch 扫描。 dirsearch -u https://1…

FPGA学习最好的2个网站?

自学FPGA最好的两个网站: Xilinx官方网站: ​网址链接&#xff1a; https://www.amd.com/zh-cn.html Xilinx Wiki - Confluence (http://atlassian.net) Xilinx GitHub&#xff08;https://github.com/Xilinx&#xff09; 电子创新网赛灵思社区 | 电子创新网 (http://eet…

原生APP开发的技术优势

原生应用程序&#xff08;Native App&#xff09;是直接使用特定操作系统的编程语言和开发工具为特定平台&#xff08;如iOS、Android等&#xff09;开发的应用程序。原生APP开发具有以下优势。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流…

深度学习推理显卡设置

深度学习推理显卡设置 进入NVIDIA控制面板&#xff0c;选择 “管理3D设置”设置 "低延时模式"为 "“超高”"设置 “电源管理模式” 为 “最高性能优先” 使用锁频来获得稳定的推理 法一&#xff1a;命令行操作 以管理员身份打开CMD查看GPU核心可用频率&…

服务器数据恢复—KVM虚拟机被误删除如何恢复虚拟磁盘文件?

服务器数据恢复环境&故障&#xff1a; 1台服务器&#xff0c;Linux操作系统EXT4文件系统&#xff0c;部署了数台KVM虚拟机&#xff0c;每台虚拟机包含一个qcow2格式的磁盘文件&#xff0c;和一个raw格式的磁盘文件。 工作人员操作失误删除了3台服务器上的KVM虚拟机&#xf…