一文带你了解所有常用排序算法

目录

快速排序

堆排序

桶排序

归并排序

拓扑排序

本文主要介绍那些我在刷题过程中常用到的排序算法: 快速排序,堆排序,桶排序,归并排序,拓扑排序

其余算法例如冒泡,插入这种效率特别低的算法就不介绍了,用的可能性极小

每一个算法都将采用例题加解释的方式进行介绍


快速排序

所谓快速排序,简单说就是枚举出一个中心点x,用双指针找出左边第一个大于等于x的元素i和右边第一个小于等于x的元素j,交换这两个元素,使得i左边维护的都是小于等于x的元素,j右边都是大于等于x的元素,再缩小范围继续排序

有点抽象是不是,那我们看一下快排的模板

void quick_sort(int q[], int l, int r)
{if (l >= r) return;int i = l - 1, j = r + 1, x = q[l + r >> 1];while (i < j){do i ++ ; while (q[i] < x);do j -- ; while (q[j] > x);if (i < j) swap(q[i], q[j]);}quick_sort(q, l, j), quick_sort(q, j + 1, r);
}

可以看出排序操作是通过swap完成的,找到不符合要求的元素后交换他们的位置,使得i左边的元素都是小于x的,而j右边的元素都是大于x的,这样我们就可以把小于x和大于x的元素分开,但是分开后元素依旧是无序的,因此我们需要递归,缩小范围直到区间内只有一个元素

而递归的边界即可以用j表示也可以用i表示,当用i表示时递归函数如下

quick_sort(q,l,i-1),quick_sort(q,i,r);

那什么时候用i什么时候用j呢,这就涉及到我们的边界问题

关于边界问题我这里引用别人的文章,里面已经说的很详细了:AcWing 785. 快速排序 - AcWing

如果不想了解这么多直接背模板就可以了,模板是可以直接套的

acwing 785.快速排序

给定你一个长度为n的整数数列。

请你使用快速排序对这个数列按照从小到大进行排序。

并将排好序的数列按顺序输出。

数据范围:
1 ≤ n ≤ 100000

参考代码:

#include<iostream>
using namespace std;
const int N = 1e6 + 10;
int n;
int q[N];void quick_sort(int q[], int l, int r)
{if (l>=r) return;int x = q[(l+r)/2], i = l - 1, j = r + 1;while(i<j){do i++; while(q[i]<x);do j--; while(q[j]>x);if (i<j) swap(q[i],q[j]);}quick_sort(q,l,j);quick_sort(q,j+1,r);
}int main()
{scanf("%d",&n);for (int i=0;i<n;i++)scanf("%d",&q[i]);quick_sort(q,0,n-1);for (int i=0;i<n;i++)printf("%d ",q[i]);return 0;
}

堆排序

堆排序一般是通过priority_queue这个数据结构来实现的,通过优先队列实现大顶堆小顶堆,用起来是极其方便的,而且排序效率高,时间复杂度为nlogn,唯一的缺点就是不好获取队列中间的元素

priority_queue<int,vector<int>,less<int>>q;//小顶堆
priority_queue<int,vector<int>,greater<int>>q;//大顶堆,不填第三个参数默认是大顶堆

优先队列的第三个参数是比较器,我们可以传入我们自定义的比较函数,也可以使用内置的表达方式如less<int>,表示整数越小的优先级越高

leetcode 347.前 K 个高频元素

347. 前 K 个高频元素 - 力扣(LeetCode)

给你一个整数数组 nums 和一个整数 k ,请你返回其中出现频率前 k 高的元素。你可以按 任意顺序 返回答案。

利用优先队列维护一个大小为k的小顶堆即可,参考代码如下:

class Solution {
public:static bool cmp(pair<int,int>&n,pair<int,int>&m){return n.second>m.second;}vector<int> topKFrequent(vector<int>& nums, int k)  {unordered_map<int,int>umap;for(auto &p:nums){umap[p]++;}priority_queue<pair<int,int>,vector<pair<int,int>>,decltype(&cmp)>q(cmp);for(auto &a:umap){q.push(a);if(q.size()>k)q.pop();}vector<int>ans;while(!q.empty()){ans.emplace_back(q.top().first);q.pop();}return ans;}
};

关于优先队列的比较器,我在我的题解中有详细介绍:. - 力扣(LeetCode)

简单来说就是如果是内置数据类型可以使用内置的比较函数(如less,greater),如果是自定义类型则需要自定义比较函数,传入自定义函数有两种方式:1.将比较函数写在结构体中 2.用decltype获取函数类型

有兴趣的同学可以做一下 leetcode 295. 数据流的中位数 这题,考察的是大小顶堆的使用

桶排序

桶排序有点类似于我们的哈希表,原理就是将元素一一映射到数组的一个位置上,每一个元素的映射规则都相同,映射结果唯一

f(x1)=y1 f(x2)=y2           

有且仅有x1==x2时才有y1==y2

桶排序的映射函数并不复杂,往往是在原索引的基础上加上一个偏移量,在数组中完成映射

例如0<=nums[i]<100那么我们就可以通过nums[i]的大小确定映射到哪个桶,因为nums[i]<100,所以我们用大于等于100的数去偏移就不会产生冲突nums[i]->bucket[nums[i]+100]

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

215. 数组中的第K个最大元素 - 力扣(LeetCode)

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

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

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

其中 -104 <= nums[i] <= 104

看到这个数据范围,我们可以发现这题可以使用桶排序,将nums[i]的值作为索引再加一个偏移量即可

参考代码:

class Solution {
public:int findKthLargest(vector<int>& nums, int k) {vector<int>bucket(21001,0);for(int i=0;i<nums.size();i++){bucket[nums[i]+10000]++;}int cnt=0;for(int i=21000;i>=0;i--){cnt+=bucket[i];if(cnt>=k){return i-10000;}}return -1;}
};

归并排序

归并排序和快速排序的思想有点类似,都是用递归实现,只不过归并注重的是归,快排注重的是递

先看模板

void merge_sort(int q[], int l, int r)
{if (l >= r) return;int mid = l + r >> 1;merge_sort(q, l, mid);merge_sort(q, mid + 1, r);int k = 0, i = l, j = mid + 1;while (i <= mid && j <= r)if (q[i] <= q[j]) tmp[k ++ ] = q[i ++ ];else tmp[k ++ ] = q[j ++ ];while (i <= mid) tmp[k ++ ] = q[i ++ ];while (j <= r) tmp[k ++ ] = q[j ++ ];for (i = l, j = 0; i <= r; i ++, j ++ ) q[i] = tmp[j];
}

思路非常简单:就是将数组分为两部分,分别对两部分进行有序化处理,有序化之后通过两两比较将两个数组合并成一个数组,再将其赋值给原数组

leetcode 148.排序链表

148. 排序链表 - 力扣(LeetCode)

给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表  

如果要求原地排序的话其实这题并不好想,当排序用在链表这种结构上时,逻辑要求会更高,但是总体思路是不变的:一分为二,分别有序化,最后再两两比较合并链表

参考代码:

class Solution {
private:ListNode* merge(ListNode*l1,ListNode*l2){ListNode*dummy=new ListNode(0);//虚拟节点ListNode*ans=dummy;while(l1&&l2){//排序auto &node=l1->val<l2->val?l1:l2;dummy->next=node;dummy=dummy->next;node=node->next;}//多余的接到后面if(l1)dummy->next=l1;else dummy->next=l2;return ans->next;}
public:ListNode* sortList(ListNode* head,ListNode*tail=nullptr) {if(head==nullptr)return head;if(head->next==tail)//只剩两个节点时{head->next=nullptr;return head;}ListNode*fast=head;ListNode*slow=head;while(fast!=tail&&fast->next!=tail){slow=slow->next;fast=fast->next->next;}ListNode*mid=slow;//找到中间节点,将链表分成两份return merge(sortList(head,mid),sortList(mid,tail));}
};

拓扑排序

拓扑排序:对有向无环图(DAG)中的所有顶点进行线性排序,使得对于任意一对顶点u和v,如果图中存在一条从u指向v的有向边,那么在拓扑序列中u出现在v之前

一句话说就是只有入度为0的点才能被选择,通过删除边(减少入度)实现节点的排序

最经典的拓扑排序就是先修课问题

 leetcode 207. 课程表

207. 课程表 - 力扣(LeetCode)

你这个学期必须选修 numCourses 门课程,记为 0 到 numCourses - 1 。

在选修某些课程之前需要一些先修课程。 先修课程按数组 prerequisites 给出,其中 prerequisites[i] = [ai, bi] ,表示如果要学习课程 ai 则 必须 先学习课程  bi 。

例如,先修课程对 [0, 1] 表示:想要学习课程 0 ,你需要先完成课程 1 。

请你判断是否可能完成所有课程的学习?如果可以,返回 true ;否则,返回 false 。

 入度数组数组记录每一门课的入度多少,通过bfs队列实现

参考代码:

class Solution {
public:bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {int hasLearn=0;queue<int>q;vector<int>indegree(numCourses,0);vector<vector<int>>gragh(numCourses);for(int i=0;i<prerequisites.size();i++){indegree[prerequisites[i][0]]++;gragh[prerequisites[i][1]].push_back(prerequisites[i][0]);}//初始化for(int i=0;i<numCourses;i++){if(indegree[i]==0)q.push(i);}//插入入度为0的课while(!q.empty()){int index=q.front();q.pop();hasLearn++;for(auto p:gragh[index])//减少需要当前课的课的入度{--indegree[p];if(indegree[p]==0){q.push(p);}}}return hasLearn==numCourses;}
};

写在末尾

跟着做完这几题可以说是对排序算法入门了,后面要做的就是多练习多总结

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

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

相关文章

C++语法|虚函数与多态详细讲解系列(包含多重继承内容)

这几天把一些书籍课程&#xff0c;还有一些个人感悟整理了一个合集&#xff0c;算是把C面向对象编程中最让人感到疑惑的内容做了一个总结&#xff0c;文章导读如下&#xff1a; C语法&#xff5c;虚函数与多态详细讲解&#xff08;一&#xff09;&#xff5c;再谈构造函数&…

中科大6系+先研院+中南大学电子信息学院2023年保研经历

中科大6系 英语口语问题&#xff1a; What’s your research plan&#xff1f;Please introduce your project. 专业课问题&#xff1a; BPSK和QPSK每个字母代表的含义&#xff1f;QAM的星座图是什么样的&#xff1f;根据什么准则画成那个样子&#xff1f; 中科大先研院 …

行业首发 | MS08067-SecGPT(送邀请码)

一、简介 MS08067-SecGPT基于LLM大模型技术专门为网络安全领域设计的智能助手&#xff0c;集问答、分析、工具为一体的对话式安全专家&#xff0c;支持可以创建多会话问答。目的是辅助用户完成网络安全相关的工作&#xff0c;学员通过问答方式体验到SecGPT所具备的威胁情报分…

flume使用实例

1、监听端口a1.sources.r1.type netcat 配置文件nc-flume-console.conf # Name the components on this agent a1 表示jvm进程名 a1.sources r1 a1.sinks k1 a1.channels c1 # Describe/configure the source a1.sources.r1.type netcat a1.sources.r1.bind node…

香橙派——自发热点连Wi-Fi

文章目录 1. 安装依赖2. 热点功能&#xff08;1&#xff09;开启热点&#xff08;并断开当前连接的Wi-Fi&#xff09;&#xff08;2&#xff09;运行Web后端 3. 开启自启动脚本方式一&#xff1a;配置系统开机自启动service服务 其他常用命令及脚本 1. 安装依赖 sudo apt-get …

VERY DEEP CONVOLUTIONAL NETWORKS FOR LARGE-SCALE IMAGE RECOGNITION--论文笔记

论文笔记 论文来源 Very Deep Convolutional Networks for Large-Scale Image Recognition 代码来源 还没上传 数据集 这里采用的是猫狗数据集 还没上传 1论文摘要的翻译 在这项工作中&#xff0c;我们研究了卷积网络深度对其在大规模图像识别设置中的准确性的影响。我…

uniapp 蓝牙分包发送数据

// 连接蓝牙设备并发送数据 function sendDataOverBLE(deviceId, data) {// 首先连接蓝牙设备uni.createBLEConnection({deviceId: deviceId,success: function(res) {// 连接成功后&#xff0c;获取服务uni.getBLEDeviceServices({deviceId: deviceId,success: function(res) …

CSP模板生成系统

开门见山&#xff0c;直接放一百分代码&#xff1a; #include<iostream> #include<algorithm> #include<cstring> #include<bits/stdc.h> using namespace std; const int N 2e4 10; string a[1010]; unordered_map<string , string> mp;void…

防火墙技术基础篇:解析入侵检测与预防系统(IDPS)功能

防火墙技术基础篇&#xff1a;解析入侵检测与预防系统&#xff08;IDPS&#xff09;功能 入侵检测与预防系统&#xff08;Intrusion Detection and Prevention Systems, IDPS&#xff09;作为防火墙技术的核心组成部分&#xff0c;扮演着保护网络安全的关键角色。本文将全面讲…

mysqlbinlog解析

安装mysql后在bin文件夹下搜索mysqlbinlog.exe&#xff0c;用cmd执行以下命令 mysqlbinlog --no-defaults --base64-outputdecode-rows -vv C:\Users\Desktop\mysql-bin.001914 > C:\Users\Desktop\output14.sql–no-defaults&#xff1a;不要读取任何选项文件。如果由于从…

【力扣一轮】344.反转字符串 541.反转字符串Ⅱ

344.反转字符串 思路 两个指针向中间移动&#xff0c;反转字符串即可。 伪代码 遍历字符串两个指针分别指向开头及末尾交换开头及末尾元素代码 void reverseString(string & s) {for(int i 0, j s.size()-1 ; i < s.size()/2 ; i , j--){char temp s[i];s[i]s…

OSPF状态机及网络接口类型

、OSPF 状态机 Down一旦接收到hello 包进人下一个状态机 Init 初始化接收到的hello 包中&#xff0c;若存在本地的 RID&#xff0c;进入下一状态 2way 双向通讯--邻居关系建立的标志 条件匹配:点到点网络直接进入下一个状态机 MA 网络将进行 DR/BDR 选举(40S) 非 DR…

一个程序员的牢狱生涯(34)利害

星期一 利害 老杨让我把煤气灶上的炒瓢里倒了一瓢水后,又让我帮他打开面粉袋,然后一边从里面抓出一把面粉放到水瓢里,一边和我说道: “我也怀疑是小X州,但当时你并没有证据能证明是他放的,对吧!” 我点点头后,老杨继续说道: “不要看别人好像在号子里想起来就能欺负他…

东北师范大学信息科学与技术学院课程攻略 共享计划

文章目录 介绍前言特性平台为什么采用 GitHub 项目作为平台&#xff1f;上传什么资料&#xff1f;警告 相关说明命名说明Watch、Star、Fork的说明 使用指南查找搜索文件下载方式贡献方法 开源力量许可附录东北师范大学学风建设实施细则第六章 规范与惩处&#xff08;学生&#…

[力扣题解] 841. 钥匙和房间

题目&#xff1a;841. 钥匙和房间 思路 深度优先搜索&#xff0c;只从0号房间进入&#xff0c;最后统计遍历情况&#xff0c;如果有没有遍历到的&#xff0c;返回false&#xff0c;否则返回true&#xff1b; 代码 class Solution { private:void dfs(vector<vector<i…

蓝桥杯物联网竞赛_STM32L071KBU6_关于TIM的新理解

最近在复习kbu6的模块&#xff0c;其中关于定时器的记忆比较模糊&#xff0c;再来强化强化 仔细看了看相关HAL库代码果然有一些额外的收获 对定时器tim7: 这个定时器还是比较准的&#xff0c;其调用方法如下&#xff1a; HAL_TIM_Base_Start_IT(&htim7);从上述代码中不难…

解决 Redis 击穿问题之黑名单

以商品列表为例&#xff1a; 主要思想&#xff1a; /* 1.先查黑名单是否存在当前商品&#xff0c;如果有就抛出异常 2.检查redis里面是否有该商品&#xff0c;如果有就直接返回&#xff0c; 3.如果没有就查询数据库&#xff0c;将数据缓存到redis 4.判断数据库中是否有该商品&…

哪个网盘最适合个人文件长期储存?用派盘最好

派盘是一款面向个人和企业的本地云存储解决方案,专为长期文件存储而设计。这种存储方式利用了本地硬盘的存储容量,通过“云化”的方式,可以将本地硬盘变成云存储空间。它具有强大的数据保护功能,确保了数据的私密性和安全性。 派盘的主要特点 高效的存取速度:由于使用本地…

这种电脑原来这么耗电……震惊了粉丝小姐姐

前言 在今年1月份的时候&#xff0c;一位来自重庆的小姐姐加了小白&#xff0c;咨询电脑的问题&#xff1a; 哦豁&#xff0c;这个电脑看着确实闪闪发光&#xff0c;是真的很漂亮&#xff5e;&#xff08;嗯&#xff0c;小姐姐也很漂亮&#xff09; 电脑无法开机&#xff0c;按…

什么是流量削峰?如何解决秒杀等业务的削峰场景

文章推荐 1 作为程序员&#xff0c;开发用过最好用的AI工具有哪些&#xff1f; 2 Github Copilot正版的激活成功&#xff0c;终于可以chat了 3 idea,pycharm等的ai assistant已成功激活 4 新手如何拿捏 Github Copilot AI助手&#xff0c;帮助你提高写代码效率 5 Jetbrains的a…