HTML5学堂-码匠:前几期“算法之旅”跟大家分享了冒泡排序法和选择排序法,它们都属于时间复杂度为O(n^2)的“慢”排序。今天跟大家分享多种排序算法里使用较广泛,速度快的排序算法
—— 快速排序法 [ 平均时间复杂度为O (n logn) ]。

Tips 1:关于“算法”及“排序”的基础知识,在此前“选择排序法”中已详细讲解,可点击文后的相关文章链接查看,在此不再赘述。
Tips 2:如果无特殊说明,本文的快速排序是从小到大的排序。

快速排序法的原理

快速排序是一种划分交换排序,它采用分治的策略,通常称其为分治法

分治法

基本思想:将原问题分解为若干个规模更小但结构与原问题相似的子问题。递归地解决这些子问题,然后将这些子问题的结果组合成原问题的结果。

基本原理

从序列中任选一个数作为“基准”;
所有小于“基准”的数,都挪到“基准”的左边;所有大于等于“基准”的数,都挪到“基准”的右边;
在这次移动结束之后,该“基准”就处于两个序列的中间位置,不再参与后续的排序;
针对“基准”左边和右边的两个子序列,不断重复上述步骤,直到所有子序列只剩下一个数为止。

原理图解

现有一个序列为 [8, 4, 7, 2, 0, 3, 1],如下演示快速排序法如何对其进行排序。

实现快速排序的步骤分解

选择“基准”,并将其从原始数组分离

先获取基准的索引值,再使用splice数组方法取出基准值。

Tips:该实例中, 基准的索引值 = parseInt(序列长度 / 2)
Tips:splice方法会改变原始数组。例如,arr = [1, 2, 3]; 基准索引值为1,基准值为2,原始数组变为arr = [1, 3];

遍历序列,拆分序列

与“基准”比较大小,并拆分为两个子序列
小于“基准”的数存储于leftArr数组当中,大于等于“基准”的数存储于rightArr数组当中

Tips:当然,也可以将 小于等于“基准”的数存于leftArr,大于“基准”的数存于rightArr
由于要遍历序列,将每一个数与“基准”进行大小比较,所以,需要借助for语句来实现

递归调用,遍历子序列并组合子序列的结果

定义一个函数,形参用于接收数组

function quickSort(arr) { };

实现递归调用遍历子序列,用concat数组方法组合子序列的结果

判断子序列的长度

递归调用的过程中,子序列的长度等于1时,则停止递归调用,返回当前数组。

快速排序法完整代码

快速排序法的效率

时间复杂度

最坏情况:每一次选取的“基准”都是序列中最小的数/最大的数,这种情况与冒泡排序法类似(每一次只能确定一个数[基准数]的顺序),时间复杂度为O(n^2)
最好情况:每一次选取的“基准”都是序列中最中间的一个数(是中位数,而不是位置上的中间),那么每次都把当前序列划分成了长度相等的两个子序列。这时候,第一次就有n/2、n/2两个子序列,第二次就有n/4、n/4、n/4、n/4四个子序列,依此类推,n个数一共需要logn次才能排序完成(2^x=n,x=logn),然后每次都是n的复杂度,时间复杂度为O(n logn)

空间复杂度

最坏情况:需要进行n‐1 次递归调用,其空间复杂度为 O(n)
最好情况:需要logn次递归调用,其空间复杂度为O(logn)

算法的稳定性

快速排序是一种不稳定排序算法
例如:现有序列为[1, 0, 1, 3],“基准”数字选择为第二个1
在第一轮比较之后,变成了[0, 1, 1, 3],左序列为[0],右序列为[1, 3](右序列的1是此前的第一个1)
不难发现,原序列的两个1的先后顺序被破坏了,改变了先后顺序,自然就是“不稳定”的排序算法了

关于O

在此前的“冒泡排序法”一文当中,我们详细讲解过O是什么,在此就不多说了,直接上图吧

相关文章链接

算法之旅 | 选择排序法
算法之旅 | 冒泡排序法

开开心心每一天

生活艰辛,代码不易,但,不要忘记微笑!


算法之旅 | 快速排序法相关推荐

  1. 探索初级算法学习笔记-快速排序法

    快速排序法学习笔记 #include<stdio.h>void swap(int *a,int *b) {int t;t=*a;*a=*b;*b=t; }void quickSort(in ...

  2. 《啊哈!算法》 关于快速排序法为什么一定要哨兵j 先出动的原因?

    得出结论: 如果选取最左边的数arr[left]作为基准数,那么先从右边开始可保证i,j在相遇时,相遇数是小于基准数的,交换之后temp所在位置的左边都小于temp.但先从左边开始,相遇数是大于基准数 ...

  3. 算法之旅 | 选择排序法

    算法之旅 | 选择排序法 HTML5学堂-码匠:数据快速的计算与排序,与前端页面性能有直接的关系.由于排序的算法有很多,在本次"算法系列"的分享当中,我们先从简单易上手的选择排序法 ...

  4. php四种基础算法:冒泡,选择,插入和快速排序法

    许多人都说 算法是程序的核心,一个程序的好于差,关键是这个程序算法的优劣.作为一个初级phper,虽然很少接触到算法方面的东西 .但是对于冒泡排序,插入排序,选择排序,快速排序四种基本算法,我想还是要 ...

  5. oracle快速排序法,经典算法系列之快速排序算法

    快速排序(Quicksort)是对冒泡排序的一种改进.由C. A. R. Hoare在1962年提出.在平均状况下,排序 n 个项目要Ο(n log n)次比较.在最坏状况下则需要Ο(n2)次比较,但 ...

  6. 【重温基础算法】内部排序之快速排序法

    内部排序之快速排序法 文章目录 内部排序之快速排序法 主要思想 过程演示 JAVA代码 算法分析 时间复杂度分析 最好时间复杂度 最坏时间复杂度 平均时间复杂度 空间复杂度 对冒泡排序的一种优化 主要 ...

  7. PHP冒泡排序算法和快速排序法

    算法说明: 冒泡排序大概的意思是依次比较相邻的两个数,然后根据大小做出排序,直至最后两位数.由于在排序过程中总是小数往前放,大数往后放,相当于气泡往上升,所以称作冒泡排序.但其实在实际过程中也可以根据 ...

  8. 经典算法之快速排序法(附B站最细讲解视频)

    活动地址:21天学习挑战赛 文章目录 一.算法 1.算法概述 2.算法步骤 3.算法特点 二.算法实践 1.Java代码 2.执行结果 3.讲解视频 三.复杂度分析 1.时间复杂度 2.空间复杂度 一 ...

  9. java 数据结构与算法 ——快速排序法

    快速排序法: 顾名思议,快速排序法是实践中的一种快速的排序算法,在c++或对java基本类型的排序中特别有用.它的平均运行时间是0(N log N).该算法之所以特别快,主要是由于非常精练和高度优化的 ...

最新文章

  1. ImportError: /lib64/libc.so.6: version `GLIBC_2.17‘ 问题解决
  2. 使用EHPC实现“完美并行”的高效批处理方案
  3. mysqldatareader什么意思_MySqlDataReader在Using中使用
  4. linux运维实战练习
  5. hdu 1175 连连看
  6. maven打包jar上传到nexus
  7. 【BZOJ4417】: [Shoi2013]超级跳马
  8. 信号与系统 徐亚宁 matlab程序,信号与系统(第4版)
  9. CentOS 安装最新版本 Git
  10. 利用ACS实现AAA服务的搭建
  11. tableView选中行的调用顺序/ 取消选中Cell
  12. Sql Server 当前日期第一天
  13. ASP.NET MVC diyUpload上传文件!
  14. 基于LPC2148的音频分析仪设计
  15. python编程从入门到实践练习7-10:梦想的旅游胜地
  16. NSDI 2021 Breaking the Transience-Equilibrium Nexus: A New Approach to Datacenter Packet Transport笔记
  17. 1、计算机毕业设计论文分析-班主任管理系统
  18. 名词解释 算法的有限性_数据结构复习之【数据结构和算法概念】
  19. LintCode 653: Expression Add Operators
  20. 关于Symbian 模拟器一闪就没的解决办法(Eclipse+MTJ+symbian模拟器)

热门文章

  1. C#Convert.ToInt32(bool)方法-将bool值转换为int
  2. COLLEGE.sql(复制的时候注意路径!!!)
  3. ffmpeg 命令画中画效果
  4. 折半查找和二叉排序树的时间性能_leecode刷题----二分搜索与二叉查找(排序)树...
  5. 不支持对系统目录进行即席更新_「目录」让你的文档结构一目了然
  6. linux中updatedb命令详解,updatedb命令
  7. Effective C++学习第三天
  8. task_struct解析
  9. 数据可视化【十一】树状图
  10. 通过编程解决问题的正确思路