漫画:什么是快速排序算法?
创始人
2025-05-29 20:13:22
0

这篇文章,以对话的方式,详细着讲解了快速排序以及排序排序的一些优化。

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
一禅:归并排序是一种基于分治思想的排序,处理的时候可以采取递归的方式来处理子问题。我弄个例子吧,好理解点。例如对于这个数组arr[] = { 4,1,3,2,7,5,8,0}。
在这里插入图片描述
我们把它切割成两部分。

在这里插入图片描述
把左半部分和右半部分分别排序好。

在这里插入图片描述
之后再用一个临时数组,把这两个有序的子数组汇总成一个有序的大数组
在这里插入图片描述
排好之后在复制原源arr数组
在这里插入图片描述
这时,源数组就排序完毕了

在这里插入图片描述
在这里插入图片描述
一禅:左半部分和右半部分的排序相当于一个原问题的一个子问题的,也是采取同样的方式,把左半部分分成两部分,然后…

直到分割子数组只有一个元素或0个元素时,这时子数组就是有序的了(因为只有一个元素或0个,肯定是有序的啊),就不用再分割了,直接返回就可以了(当然,我在讲解这个归并排序的过程中,是假设你大致了解归并排序的前提下的了)

在这里插入图片描述
在这里插入图片描述
一禅:把一个n个元素的数组分割成只有一个元素的数组,那么我需要切logn次,每次把两个有序的子数组汇总成一个大的有序数组,所需的时间复杂度为O(n)。所以总的时间复杂度为O(nlogn)

快速排序

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
小白:那倒不是,快速排序的平均时间复杂度也是O(nlogn),不过他不需要像归并排序那样,还需要一个临时的数组来辅助排序,这可以节省掉一些空间的消耗,而且他不像归并排序那样,把两部分有序子数组汇总到临时数组之后,还得在复制回源数组,这也可以节省掉很多时间。

在这里插入图片描述
在这里插入图片描述
小白:快速排序也是和归并排序差不多,基于分治的思想以及采取递归的方式来处理子问题。例如对于一个待排序的源数组arr = { 4,1,3,2,7,6,8}。

在这里插入图片描述
我们可以随便选一个元素,假如我们选数组的第一个元素吧,我们把这个元素称之为”主元“吧。
在这里插入图片描述
然后将大于或等于主元的元素放在右边,把小于或等于主元的元素放在左边。
在这里插入图片描述
通过这种规则的调整之后,左边的元素都小于或等于主元,右边的元素都大于或等于主元,很显然,此时主元所处的位置,是一个有序的位置,即主元已经处于排好序的位置了。

主元把数组分成了两半部分。把一个大的数组通过主元分割成两小部分的这个操作,我们也称之为分割操作(partition)。

接下来,我们通过递归的方式,对左右两部分采取同样的方式,每次选取一个主元 元素,使他处于有序的位置。
在这里插入图片描述

那什么时候递归结束呢?当然是递归到子数组只有一个元素或者0个元素了
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

分割操作:单向调整

一禅:就按照你说的,选一个主元,你刚才选的是第一个元素为主元,这次我选最后一个为主元吧,哈哈。假设数组arr的范围为[left, right],即起始下标为left,末尾下标为right。源数组如下
在这里插入图片描述
然后可以用一个下标 i 指向 left,即 i = left ;用一个下标 j 也指向l eft,即j = left
在这里插入图片描述
接下来 j 从左向右遍历,遍历的范围为 [left, right-1] ,遍历的过程中,如果遇到比主元小的元素,则把该元素与 i 指向的元素交换,并且 i = i +1
在这里插入图片描述
当j指向1时,1比4小,此时把i和j指向的元素交换,之后 i++。
在这里插入图片描述
就这样让j一直向右遍历,直到 j = right
在这里插入图片描述
遍历完成之后,把 i 指向的元素与主元进行交换,交换之后,i 左边的元素一定小于主元,而 i 右边的元素一定大于或等于主元。这样,就 i 完成了一次分割了。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
一禅一言不合就把代码撸好了,第一版代码如下:

//分割操作:方法一,单向调整
int partion(int a[], int left, int right)
{int temp,pivot;//pivot存放主元int i,j;i = left;pivot = a[right];for(j = left;j < right;j++){if(a[j] < pivot){  //交换值temp = a[i];a[i] = a[j];a[j] = temp;i++;}}a[right] = a[i];a[i] = pivot;return i;//把主元的下标返回
}
//快速排序
void QuickSort(int a[], int left, int right)
{int center;int i,j;int temp;if(left < right){center = partion(a,left,right);QuickSort(a,left,center-1);//左半部分QuickSort(a,center+1,right);//右半部分}
}

分割操作:双向调整

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
小白:对啊,因为你这调整方法,可能会出现对同一个元素,进行多次交换,例如刚才你在演示的那组元素,在j向右遍历交换的过程中:

第一次:8和1交换

第二次:8和3交换

第三次:8和2交换

8被重复交换了很多次
在这里插入图片描述
在这里插入图片描述
小白:其实,我们可以这样来调整元素。我还是用我的第一个元素充当主元吧。哈哈

源数组如下
在这里插入图片描述
然后用令变量i = left + 1,j = right。然后让 i 和 j 从数组的两边向中间扫描。
在这里插入图片描述
i 向右遍历的过程中,如果遇到大于或等于主元的元素时,则停止移动,j向左遍历的过程中,如果遇到小于或等于主元的元素则停止移动
在这里插入图片描述
当i和j都停止移动时,如果这时i < j,则交换 i, j 所指向的元素。此时 i < j,交换8和3
在这里插入图片描述
然后继续向中间遍历,直到i >= j。
在这里插入图片描述
此时i >= j,分割结束。

最后在把主元与 j 指向的元素交换(当然,与i指向的交换也行)。
在这里插入图片描述
这个时候,j 左边的元素一定小于或等于主元,而右边则大于或等于主元。

到此,分割调整完毕

代码如下:

方法二:双向扫描
int partition2( int[] arr, int left, int right)
{int pivot = arr[left];int i = left + 1;int j = right;while(true){  //向右遍历扫描while(i <= j && arr[i] <= pivot) i++;//向左遍历扫描while(i <= j && arr[j] => pivot) j--;if(i >= j)break;//交换int temp = arr[i];arr[i] = arr[j];arr[j] = temp;}//把arr[j]和主元交换arr[left] = arr[j];arr[j] = povit;return j;
}

时间复杂度

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
小白:因为快速排序的最坏时间复杂度是O(n2)。

例如有可能会出现一种极端的情况,每次分割的时候,主元左边的元素个数都为0,而右边都为n-1个。这个时候,就需要分割n次了。而每次分割整理的时间复杂度为O(n),所以最坏的时间复杂度为O(n2)。

最好的情况就是每次分割都能够从数组的中间分割了,这样分割logn次就行了,此时的时间复杂度为O(nlogn)。

而平均时间复杂度,则是假设每次主元等概率着落在数组的任意位置,最后算出来的时间复杂度为O(nlogn),至于具体的计算过程,我就不展开了。

不过显然,像那种极端的情况是极少发生的。
在这里插入图片描述
在这里插入图片描述
小白:哈哈,之所以说它快,是因为它不像归并排序那样,需要额外的辅助空间,而且在分割调整的时候,不像归并排序那样,元素还要在辅助数组与源数组之间来回复制。

稳定性

在这里插入图片描述
在这里插入图片描述
一禅:不是啊,例如,在排序的过程中,主元在和j交换的时候是有可能破坏稳定性的,例如
在这里插入图片描述
把主元与j指向的元素进行交换
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

//随机选取主元
int random_partition(int[] arr, int left, int right)
{i = random(left, right);//随机选取一个位置//在把这个位置的元素与ar[left]交换swap(arr[i], arr[left]);return partition(arr, left, right);
}

终于写完,这个快排写了挺长时间,觉得有收获的话,可以转发支持一波哦(´-ω-`)。

更多排序算法文章

1. 漫画:什么是冒泡排序算法?

2. 漫画:什么是选择排序算法?

3. 漫画:什么是插入排序算法?

4. 漫画:什么是希尔排序算法?

5. 漫画:什么是归并排序算法?

6. 漫画:什么是快速排序算法?

7. 漫画:什么是堆排序算法?

8. 漫画:什么是基数排序算法?

9. 漫画:什么是外部排序?

10. 什么是计数排序?

11. 十大排序算法极简汇总篇

推荐阅读

下载破 2w+,在校生必看,《程序员内功修炼》第二版出炉

从双非到大厂,帅地写了一本原创PDF送给大家

一个帮你拿offer的校招网站

算法刷题路线(系统+全面)

作者简介:我是帅地,校招拿到过不少大厂offer,毕业去了腾讯研发岗,毕业半年整到人生第一个 100 万,目前专注于写大学规划 + 校招求职相关的内容,著有个人原创网站 PlayOffer。

相关内容

热门资讯

企业IP打造指南:小公司低成本... 小公司做企业IP,不是为了装门面,而是让客户在没见到你之前,就能通过内容知道你是谁、你解决什么问题、...
官方:赵心童入选世界斯诺克名人... 北京时间5月8日消息,世界斯诺克巡回赛(WST)今日正式公布了2025/26赛季年终奖项及名人堂更新...
小灰熊AI学员王锋:希望能跟上... 35了,老程序员了。 从进入互联网行业到现在,其实已经做了很多年移动端开发。最早那几年,安卓行业发展...
原创 2... 2026年全国两会把稳定房地产市场列为重点工作,政府工作报告明确提出因城施策控增量、去库存、优供给。...
一年翻倍,六年未归——徽商银行... 文:向善财经 今年的港股市场,与A股市场出现了明显的分化。 A股这边,科技板块在AI浪潮中热闹非凡;...
古井贡酒2025:在行业深度调... 以“稳”为底、以“新”为翼。 文/每日财报 杜康 在行业库存高企、价格倒挂的背景下,当多数酒企在为...
好上好8408万收购鼎瑞芯加码... 5月7日晚,好上好(001298.SZ)抛出一份收购公告,拟以8408万元现金收购深圳市鼎瑞芯科技有...
全面大撤离!李嘉诚英国“套现”... 突发,李嘉诚又卖了。 这次,套现了455亿。 金额不少,但更值得关注的是透露着不同寻常的信号。 因为...
油气价格上涨加剧法国一季度贸易... 据新华社,法国海关7日发布的数据显示,受中东局势推高国际油气价格影响,法国今年第一季度贸易逆差扩大至...
昆仑芯启动科创板IPO上市辅导... 5月8日,据证监会官网显示,昆仑芯(北京)科技股份有限公司于2026年5月7日正式启动科创板上市辅导...
贵州茅台酒股份有限公司关于回购... 来源:上海证券报 证券代码:600519 证券简称:贵州茅台 公告编号:临2026-016 贵州茅...
百度昆仑芯启动科创板上市辅导,... 5月8日,证监会官网显示,昆仑芯(北京)科技股份有限公司 (下称“昆仑芯”)于2026年5月7日正式...
滕州信华的承压时刻:罚单、失信... 2026年4月末,滕州信华美元债单日跌近2%,关联方被列“老赖”。半年前,这家AA+城投曾因非市场化...
002808,或被终止上市! 【导读】因触及财务类退市指标,*ST恒久或被终止上市 中国基金报记者 李智 又一A股或被终止上市。 ...
院士团队掌舵,溧阳这家企业已完... 近日,溧阳天目先导电池材料科技有限公司(下称“天目先导”)官宣完成B轮融资,投资方包括知卓创新资本、...
工商银行全新推出“工盈研选”品... 深圳商报·读创客户端记者 詹钰叶 近日,工商银行重磅推出「工盈研选」基金销售服务品牌,以客户盈利为核...
和讯信息胡云龙:逼空走势,周五... 今天市场出现逼空走势,场内投资者因持有筹码而尤为受益。五一前布局的投资者当前收获颇丰。然而,随着上证...
今晚,油价上调! 4月21日国内成品油价格下调以来,国际市场原油价格剧烈震荡,前期大幅上涨后近日有所回落,本次调价的前...
南方东英旗下两倍做多海力士,成... 【导读】南方东英旗下两倍做多海力士,成为全球最大的个股杠杆及反向产品 中国基金报记者 伊万 人工智能...
原创 金... 黄金,这东西从古至今就没离开过中国人的生活。从老辈人压箱底的小黄鱼,到如今年轻人结婚绕不开的“三金”...