快速排序(三)——hoare法

fe594ea5bf754ddbb223a54d8fb1e7bc.gif

目录

​一.前言

二.快速排序

hoare排法​

三.结语


8fb442646f144d8daecdd2b61ec78ecd.png一.前言

本文给大家带来的是快速排序,快速排序是一种很强大的排序方法,相信大家在学习完后一定会有所收获。

码字不易,希望大家多多支持我呀!(三连+关注,你是我滴神!)

二.快速排序

快速排序是Hoare与1962年提出的一种二叉树结构的交换排序方法,其基本思想为:任取待排序元素排序中的某元素作为基准值,按照该排序码将待排序集合分割成两子序列,左子序列中所有元素均小于基准值,右子序列中所有元素均大于基准值,然后最左右子序列重复该过程,直到所有元素都排列在相应位置上为止。

上述为快速排序递归实现的主框架,发现于二叉树前序遍历规则非常像,同学们在写递归框架时可想想二叉树前序遍历规则即可快速写出来,后序只需分析如何按照基准值来对区间中数据进行划分的方式即可。 

今天我们来学习的是第一种版本:

hoare排法

下面是动态图例: 

开始解析: 

简单点讲就是我们找到一个数成为key,然后从右边出发找到比key小的数(如5)

然后左边再出发找比key大的数(如7)

然后让这两个值交换,意义是把比key小的值尽量放左边,比key大的值尽量放右边

交换完之后呢,右边再继续找小(如4)

左边也继续找大(如9)

然后两者再进行交换

再继续找小(如3)

再继续找大,但没找到反而相遇了,那就停下

然后最让key与相遇的位置交换

最后我们发现比key小的都在其左边,比key大的都在右边了。

右边找比key小的值找到后停下换左边找比key大的值然后也停下最后二者交换,直到key到达最终位置。

所以单趟的意义就是使key到达正确(排好序要放的位置)

老规矩,我们先来分析一下单趟排序代码:

那不妨想一想如果key左边5个数有序,右边4个数也有序,那么就完成排序的目的了。

而这又与我们之前学习的二叉树遍历很像,根,左子树,右子树遍历,再对左子树进行分割根,左子树,右子树遍历——前序遍历。

当我们把这个想象成二叉树分治遍历,那么就是排序全部完成的时候了。

我们可以快速来一遍单趟,设3为key,然后右边找小(2),左边找大(没找到相遇了),与key交换。 

3不用动了,再分割出左边选一个key出来。 

​ 

继续右边找小(找不到)交换。 

​ 

我们把它看成二叉树,当排好最后一组时开始往回递归,遇到key为2的一组时再往右递进,发现是空子树回归,然后继续往上到key为3的一组,对其右子树(5 4 )继续递进。

至此,左边排序已经排好了。 

​ 

 这样对右子树(6右边的排序)持续下去结束后,整个数组的排序完成。

 接下来是代码部分:

int PartSort(int* a, int left, int right)
{int key = a[left];while (left < right){//找小while (a[right] > key){right--;}//找大while (a[left] < key){left++;}Swap(&a[right], &a[left]);}Swap(key, &a[left]);return left;
}

我们定好key下标,首先当left与right相遇的时候(left==right)才会让key交换,所以我们第一层循环用的是left<right。

然后是找大和找小我们第二层循环就正常比较大小++和--就行了。

我们作好大体框架再从细节处出发(找bug):

当我们修改数组中的2个数字再次排序时。

我们会发现left与right都会在6这个位置停下,这样造成的结果就是死循环!

所以我们需要修改条件

而在我们处理好上面这个问题后又会出现新问题:数组越界 

可以发现如果是如图中数组,那么right就会不断--移出数组外造成越界问题。

所以需要添加条件(让right--时遇到left就停下,避免越界),left同理。 

int PartSort(int* a, int left, int right)
{int key = a[left];while (left < right){//找小while (left<right && a[right] >= key){right--;}//找大while (left < right && a[left] <= key){left++;}Swap(&a[right], &a[left]);}Swap(&key, &a[left]);return left;
}

还有一个问题:当key发生交换的时候只是数值发生了交换,但key还是在原来的位置,所以我们需要把它移动到交换后的位置。

这样就可以

int PartSort(int* a, int left, int right)
{int keyi = left;while (left < right){//找小while (left<right && a[right] >= a[keyi]){right--;}//找大while (left < right && a[left] <= a[keyi]){left++;}Swap(&a[right], &a[left]);}Swap(&a[keyi], &a[left]);return left;
}

接下来就是处理分治问题: 

void QuickSort(int* a, int begin, int end)
{int keyi = PartSort(a, begin, end);//[begin,keyi-1]key[keyi+1,end]QuickSort(a, begin, keyi - 1);QuickSort(a, keyi + 1, end);}

然后我们需要制定一个结束的条件:

  • 只有一个数的时候(left==right)结束
  • 没有数的时候(left>right)

void QuickSort(int* a, int begin, int end)
{if (begin >= end){return;}int keyi = PartSort(a, begin, end);//[begin,keyi-1]key[keyi+1,end]QuickSort(a, begin, keyi - 1);QuickSort(a, keyi + 1, end);}

下面这是递归展开图: 

我们来用100万个随机数来测试一下快排的性能

可以看到快排的效率是名不虚传的~

在我们写完快排后再来回顾几个问题:

为什么相遇的位置一定比key小呢?

如果相遇的位置比key大,那交换肯定是会出问题的。

我们重新按原位置开始走,当快要相遇时,在R先走的情况下 ,能让R停下的是比key小的3。这样是让L走然后与R相遇。验证了R先走,相遇值比key小。

如果一开始是L先走,走到同样情景时,因为是L先走它会去找比key大的数,就这样找到了9,也与R相遇,但这样最后交换是错误的,相遇的位置比key大。

我们再换一种情况,把3换成10:

我们会发现如果是R先走,那么它会找小,最后越过10找到了4并与L相遇.因为L的位置一定是比key小的数字,毕竟它下标对应的数字是由R(负责找比key小的数字)找到并交换过来的。验证了R先走,相遇值比key小。

如果是L先走呢?在10停下后等R相遇然后交换,最后发现交换是错误的,因为出现了左边(10)比key(6)大的情况,相遇值比key大。

最后是一种极端情况:在几乎是升序的数组里R从右边先走直到和L相遇,相遇的位置没有比key小。交换后对其右边的一组数值再进行分治划分,

​ 

经过这几种情况分析我们可以发现,如果是L先走然后相遇值都是比key大的,并且交换都会出现错误。而在R先走然后相遇值都是比key小的,并且交换不会出现错误。

相信大家应该发现了,key在左边的时候我们就让右边先走,key在右边的时候我们让左边先走。

因为当key在左边的时候我们要确保最后的相遇值是比key小的,这样交换过来才能符合升序的规则,所以我们让R先走确保它找到的值一定是小的。同理key在右边时我们要确保交换过来的相遇值要比key大,这样才能符合升序规则,而让L先走就一定能确保它找到的是比key大的值。

最终我们需要学会根据key的位置不同,升序降序的规则不同合理作出相应的变化~

下面我们来分析快排的第二个问题:快排的效率分析

假设我们每一次选出的key都是中位数就会呈现出这种情况

我们可以看到每一层的单趟排序其实都可以看作是N次执行(在数很多的情况下),因为每一层合计起来也差不多是N这个量级。

而它的高度是logN,这样它的总的时间复杂度度就是O(N*logN)

但这只是比较理想的情况下,如果是在接近有序的情况下,它的高度就会变成N,这样时间复杂度的就会是O(N^2)

为了避免快排在有序的情况下效率受到干扰,我们设置了一个叫三数取中的方法。(不是位置取中,而是数值取中)

改变选key的策略,不再是固定选左边的值作key,但如果是中间的值作key又是先走左边还是右边呢,这样也会影响到单趟排序。其实我们可以一直选左边的值作key,就算你选到的key在中间把它换到左边就行了。

这样无论是有序还是无序最终key的交换落点都能尽量落到与下图差不多的位置,避免了有序时算法效率的损耗。

最终代码: 

int GetMidi(int* a, int left, int right)
{int mid = (left + right) / 2;// left mid rightif (a[left] < a[mid]){if (a[mid] < a[right]){return mid;}else if (a[left] > a[right])  // mid是最大值{return left;}else{return right;}}else // a[left] > a[mid]{if (a[mid] > a[right]){return mid;}else if (a[left] < a[right]) // mid是最小{return left;}else{return right;}}
}int PartSort(int* a, int left, int right)
{int midi = GetMidi(a,left,right);Swap(&a[midi], &a[left]);int keyi = left;while (left < right){//找小while (left<right && a[right] >= a[keyi]){right--;}//找大while (left < right && a[left] <= a[keyi]){left++;}Swap(&a[right], &a[left]);}Swap(&a[keyi], &a[left]);return left;
}void QuickSort(int* a, int begin, int end)
{if (begin >= end){return;}int keyi = PartSort(a, begin, end);//[begin,keyi-1]key[keyi+1,end]QuickSort(a, begin, keyi - 1);QuickSort(a, keyi + 1, end);}

4b12323f94834afd9ec146a3c10df229.jpeg三.结语

本次我们介绍了hoare的快排法,相信大家都发现了有很多的坑点需要我们注意,不过放心,下一篇文章我会介绍在原基础上优化更加的其他快排法~最后感谢大家的观看,友友们能够学习到新的知识是额滴荣幸,期待我们下次相见~

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

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

相关文章

Leetcode刷题笔记题解(C++):LCR 174. 寻找二叉搜索树中的目标节点

思路&#xff1a;二叉搜索树的中序遍历是有序的从大到小的&#xff0c;故得出中序遍历的结果&#xff0c;即要第cnt大的数为倒数第cnt的数 /*** Definition for a binary tree node.* struct TreeNode {* int val;* TreeNode *left;* TreeNode *right;* TreeN…

Maven 打包时,依赖配置正确,但是类引入出现错误,一般是快照(Snapshot)依赖拉取策略问题

问题描述&#xff1a; 项目打包时&#xff0c;类缺少依赖&#xff0c;操作 pom.xml -> Maven -> Reload project &#xff0c;还是不生效&#xff0c;但是同事&#xff08;别人&#xff09;那里正常。 问题出现的环境&#xff1a; 可能项目是多模块项目&#xff0c;结构…

Mybatis 动态SQL(set)

我们先用XML的方式实现 : 把 id 为 13 的那一行的 username 改为 ip 创建一个接口 UserInfo2Mapper ,然后在接口中声明该方法 package com.example.mybatisdemo.mapper; import com.example.mybatisdemo.model.UserInfo; import org.apache.ibatis.annotations.*; import jav…

【TOP顶刊】众望所归!这本IEEE(trans),号称行业白月光,上涨至中科院1区,评职神刊好投吗?

计算机类 • 神级顶刊 今天带来IEEE旗下trans系列顶刊&#xff0c;在最新的2023中科院分期中上涨至中科院1区&#xff0c;期刊质量实力强劲&#xff0c;如有投稿意向可重点关注&#xff0c;具体详情见下文&#xff1a; 01 期刊简介 IEEE Transactions on Automatic Control …

CRM是什么软件?

阅读本文&#xff0c;你将了解&#xff1a;一、CRM是什么软件&#xff1b;二、CRM软件如何进行客户管理&#xff1b;三、CRM软件有什么价值&#xff1b;四、优秀案例分享。 一、CRM是什么软件 CRM&#xff08;Customer Relationship Management&#xff0c;客户关系管理&…

【vscode】远程资源管理器自动登录服务器保姆级教程

远程资源管理器自动登录服务器 介绍如何配置本地生成rsa服务端添加rsa.pub配置config文件 介绍 vscode SSH 保存密码自动登录服务器 对比通过账号密码登录&#xff0c;自动连接能节约更多时间效率&#xff0c;且通过vim修改不容易发现一些换行或者引号导致的错误&#xff0c;v…

【Spring 篇】MyBatis多表操作:编织数据的交响乐

欢迎来到MyBatis的多表操作世界&#xff01;在这个充满交响乐的舞台上&#xff0c;我们将探索如何巧妙地编织多个数据表的数据&#xff0c;创造出一场旋律动听的数据交响曲。无需繁琐的SQL拼接&#xff0c;MyBatis让多表操作变得优雅而简单。让我们一起进入这个音乐殿堂&#x…

8.3 Springboot整合Redis 之Jedis方式

文章目录 前言一、Maven依赖二、新增子Module:tg-book-redis三、Jedis配置类3.1 Jedis连接池核心配置说明四、Jedis 工具类五、新增controller测试前言 Jedis是Redis官方推荐的Java客户端连接工具,用法非常简单,Jedis的API与Redis的API可以说是一模一样,所以非常有利于熟悉…

[足式机器人]Part2 Dr. CAN学习笔记- 最优控制Optimal Control Ch07-2 动态规划 Dynamic Programming

本文仅供学习使用 本文参考&#xff1a; B站&#xff1a;DR_CAN Dr. CAN学习笔记 - 最优控制Optimal Control Ch07-2 动态规划 Dynamic Programming 1. 基本概念2. 代码详解3. 简单一维案例 1. 基本概念 Richoard Bell man 最优化理论&#xff1a; An optimal policy has the …

低温恒温器的常见毛病及维护

低温恒温器广泛应用于恒温精度要求较高的科研&#xff0c;高等院校、电子仪表、生物工程、医药、固态与液态化合物的低温恒温培育。因为各种原因在运用的过程中偶然会碰到恒温水槽频繁呈现毛病的状况&#xff0c;那么低温水槽呈现毛病到底是什么原因形成的呢&#xff1f;总结下…

Linux:使用for+find查找文件并cp到其他目录,文件名带有空格

一、场景描述 在终端窗口中&#xff0c;用shell命令&#xff0c;批量拷贝文件到指定目录。 我是在Windows系统上&#xff0c;通过git bash终端来执行shell命令的。 二、实现过程 命令1 for filepath in find /d/LearningMaterials/数学/数学/高中/一数/偏基础&#xff08;基…

2.上传图片到Minio服务中

上传图片 界面原型 第一步: 用户在课程信息编辑界面可以上传课程图片或者修改上传的课程图片 第二步: 请求媒资管理服务将课程图片上传至分布式文件系统同时在媒资管理数据库保存文件信息,上传成功后返回图片在MinIO中的地址 第三步: 请求内容管理服务保存课程信息含课程封…