数据结构 快速排序

  快速排序是对冒泡排序的一种改进,是所有内部排序算法中平均性能最优的排序算法。其基本思想是基于分治法的:在待排序数组L[1...n]中任取一个元素pivot作为基准,从数组的两端开始扫描。设两个指示标志(low指向起始位置,high指向末尾),先从后向前扫描(high递减),如果high位置的元素小于pivot,就交换low和high位置的元素,然后从前向后扫描(low递增),如果low位置的元素大于pivot,就交换low和high位置的元素。重复上述过程,直到low>=high,然后把基准放到low位置上,一趟排序就完成了,将一个元素(基准)放到了最终位置上,不产生有序子序列。将待排序数组划分为两部分即L[1...k-1]和L[k+1...n],使得前半部分L[1...k-1]所有元素小于pivot,后半部分L[k+1...n]所有元素大于或等于pivot。接着采用递归的方式分别对前半部分和后半部分排序,直到每部分只有一个元素或空为止,即所有元素放在了最终位置上。

  之所以快速排序比较快,是因为相比于冒泡排序,每次交换是跳跃式的。每次排序时选取一个基准,将小于基准的数全部放到基准点的左边,将大于或等于基准的数全部放到基准的右边,在每次交换时不会像冒泡排序一样只能在相邻的数之间进行交换,增大了交换距离,减少了总的比较和交换次数,加快了速度。在最坏情况下,仍可能交换相邻的两个数。

  假设对“6 1 2 7 9 3 4 5 10 8”这10个数进行排序:

  从后向前扫描

  

  交换7和5

  

  交换之后:6 1 2 5 9 3 4 7 10 8

  交换9和4

  

  交换之后:6 1 2 5 4 3 9 7 10 8

  i>=j,交换6和3

  

  交换之后:3 1 2 5 4 6 9 7 10 8

  一趟排序结束。

 1 public class QuickSort {
 2 
 3     public static void quickSort(int[] arr, int low, int high) {
 4         // 至少有两个元素需要排序
 5         if (low < high) {
 6             int curLow = low;
 7             int curHigh = high;
 8             int temp = arr[low];
 9 
10             while(curLow < curHigh) {
11                 // 从右向左扫描,寻找第一个比基准小的数
12                 while(curLow < curHigh && arr[curHigh] >= temp) {
13                     curHigh--;
14                 }
15                 arr[curLow] = arr[curHigh];
16 
17                 // 从左向右扫描,寻找第一个比基准大的数
18                 while(curLow < curHigh && arr[curLow] <= temp) {
19                     curLow++;
20                 }
21                 arr[curHigh] = arr[curLow];
22             }
23 
24             // 基准归位
25             arr[curLow] = temp;
26 
27             // 基准位置左边子序列递归排序
28             quickSort(arr, low, curLow - 1);
29             // 基准位置左边子序列递归排序
30             quickSort(arr, curLow + 1, high);
31         }
32     }
33 
34     public static void main(String[] args) {
35         int[] arr = {15, 1, 17, 3, 6, 8};
36         QuickSort.quickSort(arr, 0, arr.length - 1);
37 
38         for (int i = 0, length = arr.length; i < length; i++) {
39             System.out.printf(" %d", arr[i]);
40         }
41     }
42 
43 }

  结果如下:

 1 3 6 8 15 17

  上述代码每次以当前数组中第一个元素作为基准对数组进行划分。

  快速排序性能分析:

  空间复杂度:因为快速排序是递归的,所以需要借助一个递归工作栈来保存每一层递归调用的必要信息,容量应与递归调用的最大深度一致。最坏情况下n-1次递归调用,栈的深度为O(n);最好和平均情况下栈的深度为O(log2n)。所以,空间复杂度在最坏情况下为O(n),平均情况下为O(log2n)。

  时间复杂度:快速排序的性能主要取决于划分操作的好坏。最坏情况下待排序数组基本有序或基本逆序时,每一次递归划分的两个区域分别包含n-1个元素和0个元素,快速排序退化成冒泡排序,时间复杂度为O(n2)。最好情况下最平衡划分,两个子数组长度不超过n/2,时间复杂度为O(nlog2n)。而快速排序平均情况下运行时间接近于最好情况下的运行时间,即时间复杂度在最坏情况下为O(n2),平均情况下为O(nlog2n)。

  稳定性:如果右端区间有两个相同的关键字,且均小于基准,那么交换到左端区间后,它们的相对次序会变化,即快速排序不稳定。例如,表L={3, 2, 2},经过一趟排序后L={2, 2, 3},最终结果是L={2, 2, 3},2与2的相对次序发生了变化。

  改进方案

  1 使用直接插入排序

  当递归过程中划分得到的子数组长度较小时,可以使用直接插入排序完成排序工作。

1 public static void quick(int []array ,int lo,int hi){
2         if(hi-lo+1<10){
3             insertSort(array);
4         }else{
5             quickSort(array,lo,hi);
6         }
7     }

  2 选取好的基准

  尽量选取可以把元素平衡划分的基准,有三种方法:固定切分,随机切分和三取样切分。固定切分的效率低。随机切分是常用的一种切分,效率高,最坏情况下时间复杂度有可能为O(n2)。三取样切分最理想,具体操作:选取数组的头尾和中间这3个元素,取这3个元素的中间值作为基准。

 1 public static int partition(int []array,int lo,int hi){
 2         //三数取中
 3         int mid=lo+(hi-lo)/2;
 4         if(array[mid]>array[hi]){
 5             swap(array[mid],array[hi]);
 6         }
 7         if(array[lo]>array[hi]){
 8             swap(array[lo],array[hi]);
 9         }
10         if(array[mid]>array[lo]){
11             swap(array[mid],array[lo]);
12         }
13         int key=array[lo];
14         
15         while(lo<hi){
16             while(array[hi]>=key&&hi>lo){
17                 hi--;
18             }
19             array[lo]=array[hi];
20             while(array[lo]<=key&&hi>lo){
21                 lo++;
22             }
23             array[hi]=array[lo];
24         }
25         array[hi]=key;
26         return hi;
27     }
28     
29     public static void swap(int a,int b){
30         int temp=a;
31         a=b;
32         b=temp;
33     }
34     public static void sort(int[] array,int lo ,int hi){
35         if(lo>=hi){
36             return ;
37         }
38         int index=partition(array,lo,hi);
39         sort(array,lo,index-1);
40         sort(array,index+1,hi);
41     }

  

  参考资料

  《2017年数据结构联考复习指导》 P287-288

  快速排序(java实现)

  坐在马桶上看算法:快速排序

转载于:https://www.cnblogs.com/WJQ2017/p/8340156.html

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

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

相关文章

小米人员架构调整:组建中国区,王川任总裁

12月13日上午&#xff0c;小米内部发布人员调整公开信&#xff0c;信中传达了两个重要内容&#xff1a;将销售与服务部改组为中国区&#xff0c;任命集团高级副总裁王川兼任中国区总裁。 在今年9月份&#xff0c;也就是小米上市前夕&#xff0c;雷军在一封内部信中宣布对公司组…

Java基础 五 方法

方法 1.1 方法概述 在我们的日常生活中&#xff0c;方法可以理解为要做某件事情&#xff0c;而采取的解决办法。 如&#xff1a;小明同学在路边准备坐车来学校学习。这就面临着一件事情&#xff08;坐车到学校这件事情&#xff09;需要解决&#xff0c;解决办法呢&#xf…

附近有什么?8款可以查周边的App

如今科技发达的时代&#xff0c;手机的功能不仅仅只是能通讯聊天&#xff0c;而是逐渐的走进了人们的生活中。因为有了APP&#xff0c;我们的生活才更丰富&#xff0c;并且有很多是我们生活中不可缺少的软件&#xff0c;而这些软件便是根据手机中的GPS定位系统而来的。简单来说…

MyEclipse小问题与汉字处理

今天在使用MyEclipse时&#xff0c;遇到工作目录报错(如上图)&#xff0c;解决方法如下&#xff1a;找到对应工作区(查看工作区的方法为&#xff1a;单击File → Switch Workspace 即可)依次打开 .metadata文件夹 → .plugins文件夹 → org.eclipse.core.runtime文件夹 → .set…

java B2B2C springmvc mybatis电子商务平台源码-消息队列之RocketMQ

RocketMQ出自阿里公司的开源产品&#xff0c;用 Java 语言实现&#xff0c;在设计时参考了 Kafka&#xff0c;并做出了自己的一些改进&#xff0c;消息可靠性上比 Kafka 更好。RocketMQ在阿里集团被广泛应用在订单&#xff0c;交易&#xff0c;充值&#xff0c;流计算&#xff…

VSCode同步设置

2022/4/1 更新 刚刚发现还有人在看这篇文章&#xff0c;这里更新一下&#xff0c;VSCode 从1.48版本开始已经内置了同步功能&#xff0c;可以不用再使用Settings Sync插件了。 点击左下角的用户或者设置的 Sign in to Sync Setting&#xff0c;使用GitHub或者Microsoft账户登…

配置三台服务器组成的ELK集群(二)

上一篇里主要是介绍了ES和ES-Head的安装过程&#xff0c;这一篇继续介绍ELK集群的其他核心组件安装过程。 五、安装Logstash&#xff1a; 本案的Logstash安装在10.113.130.117上&#xff1b;燃鹅&#xff0c;Logstash也可以利用多台组成集群&#xff0c;如果未来单台处理不过来…

Discuz X3.2源码解析 discuz_application类(转自百度)

discuz_application在/source/class/discuz/discuz_application.php中。 discuz_application继承自抽象类discuz_base discuz_application主要实现对运行环境、配置、输入、输出、数据库、设置、用户、session、移动模块、计划任务、手机预览等方面的初始化。 instance()函数来…

.NET性能优化-是时候换个序列化协议了

计算机单机性能一直受到摩尔定律的约束&#xff0c;随着移动互联网的兴趣&#xff0c;单机性能不足的瓶颈越来越明显&#xff0c;制约着整个行业的发展。不过我们虽然不能无止境的纵向扩容系统&#xff0c;但是我们可以分布式、横向的扩容系统&#xff0c;这听起来非常的美好&a…

Kubernetes-基于Helm安装部署高可用的Redis

1、Redis简介 Redis是一个开放源代码&#xff08;BSD许可证&#xff09;的代理&#xff0c;其在内存中存储数据&#xff0c;可以代理数据库、缓存和消息。它支持字符串、散列、列表、集合和位图等数据结构。Redis 是一个高性能的key-value数据库&#xff0c; 它在很大程度改进了…

markdown流程图画法小结

markdown流程图画法小结markdown画图流程图 最简单的流程图为例mermaid! graph TD A --> B //在没有(),[].{}等括号的情况之下&#xff0c;图标默认名字就是字母 A --> C C --> D B --> D 给图标添加名字&#xff0c;改变只有矩阵图形&#xff0c;在箭头上添加文字…

32岁京东毕业程序员,走投无路当了外企外包,闲得心里发慌,到点下班浑身不自在!...

‍‍当一位京东程序员进入外企当外包会怎么样&#xff1f;顺利躺平&#xff0c;实现wlb&#xff08;工作生活平衡&#xff09;吗&#xff1f;未必&#xff0c;因为人是一种很奇怪的动物。这位网友说&#xff1a;32岁京东毕业程序员&#xff0c;找了几个月工作一直没有合适的&am…

XAML 创建浏览器应用程序

XAML 创建浏览器应用程序XAML 创建浏览器应用程序作者&#xff1a;WPFDevelopersOrg - 驚鏵原文链接&#xff1a;https://learn.microsoft.com/zh-cn/dotnet/desktop/wpf/app-development/wpf-xaml-browser-applications-overview?viewnetframeworkdesktop-4.8框架使用.NET40&…

Kubernetes共享使用Ceph存储

目录 简要概述环境测试结果验证简要概述 Kubernetes pod 结合Ceph rbd块设备的使用&#xff0c;让Docker 数据存储在Ceph,重启Docker或k8s RC重新 调 度pod 不会引起数据来回迁移。 工作原理无非就是拿到ceph集群的key作为认证&#xff0c;远程rbdmap映射挂载使用。那么就要启用…

MFC界面库BCGControlBar v25.3新版亮点:Dialogs和Forms

2019独角兽企业重金招聘Python工程师标准>>> 亲爱的BCGSoft用户&#xff0c;我们非常高兴地宣布BCGControlBar Professional for MFC和BCGSuite for MFC v25.3正式发布&#xff01;新版本添加了对Visual Studio 2017的支持、增强对Windows 10的支持等。接下来几篇文…

基于 .NET 7 的 QUIC 实现 Echo 服务

前言随着今年6月份的 HTTP/3 协议的正式发布&#xff0c;它背后的网络传输协议 QUIC&#xff0c;凭借其高效的传输效率和多路并发的能力&#xff0c;也大概率会取代我们熟悉的使用了几十年的 TCP&#xff0c;成为互联网的下一代标准传输协议。在去年 .NET 6 发布的时候&#xf…

关于信息收集和加工的思考

随着互联网的发展&#xff0c;获取信息的手段越来越多&#xff0c;我们对手机的依赖程度超乎想象&#xff0c;每天忙碌着&#xff0c;大脑接收着丰富的信息&#xff0c;感觉每天都学习到了很多的知识。但我们对学习经常会有些误区&#xff1a;1、书买了摆在书架上&#xff0c;看…

8支团队正在努力构建下一代Ethereum

“我们不想在构建 Ethereum 2.0时重新造轮子。” 谈到开发人员为 Ethereum 区块链进行两个独立的升级&#xff08;一个称为 Ethereum 2.0&#xff0c;另一个称为 Ethereum 1x&#xff09;所作出的补充努力&#xff0c;劳尔乔丹坚持认为&#xff0c;在较短的时间内将升级包括在 …

windows环境下TP5.1使用think-worker(Workerman/GatewayWorker)

文章目录首先是解决如何运行gatewayworker调试gatewayworker程序向指定客户端发送消息在TP框架中调用Gateway的API总结说明测试环境 windows10&#xff1b;PHP7.2&#xff1b;TP5.1&#xff1b; 这里只介绍如何使用TP集成的workerman扩展库think-worker&#xff0c;原生workerm…

webpack之DefinePlugin使用

DefinePlugin是webpack注入全局变量的插件&#xff0c;通常使用该插件来判别代码运行的环境变量。在使用该插件需要注意的是&#xff0c;如果在该插件配置了相关的参数&#xff0c;必须要源码中使用&#xff0c;webpack才会注入。例如&#xff1a; new webpack.DefinePlugin({p…