排序算法(3):

news/2025/2/27 7:17:25

这是我们的最后一篇算法>排序算法了,也是我们的初阶数据结构的最后一篇了。

我们来看,我们之前已经讲完了插入排序,选择排序,交换排序,我们还剩下最后一个归并排序,我们今天就讲解归并排序,另外我们还要再讲解一下计数排序

我们前面的七种算法>排序算法,在排序的过程中都要进行数据大小的比较,我们把这些算法统称为比较排序。

但是我们的计数排序不是比较排序。

我们先来看我们的归并排序:

归并排序:

归并算法>排序算法思想:

归并排序(MERGE-SORT)是建⽴在归并操作上的⼀种有效的算法>排序算法,该算法是采⽤分治法(Divide andConquer)的⼀个⾮常典型的应⽤。将已有序的⼦序列合并,得到完全有序的序列;即先使每个 ⼦序列有序,再使⼦序列段间有序。若将两个有序表合并成⼀个有序表,称为⼆路归并。归并排序核 ⼼步骤:

我们看我们的这个图片,这个就相当于是我们的思路,首先,我们的这个数组是无序的,我们先把数组从中间分开,分成两个数组,然后就得到了两个数组,这两个数组也是无序的,然后我们就继续进行分组,然后我们就得到了4个无序的数组,然后我们再进行分组,最后我们就得得到了8个数组,这8个数组里面的都是只有一个数据的,所以他们就都是有序的,这时候我们就要把这些有序的数组合成一个有序的数组,我们之前学过把两个有序的数组合并成一个有序的数组,我们就把两个有序的数组合成一个,我们不断地两两相合并,直到把我们分出来的数组全部合并起来,最后我们就得到了我们的有序的数组。

接下来我们来实现一下我们的代码:

我们来看我们的代码,我们先来看下面的我们的归并排序:

我们把数组传过来,然后我们动态开辟一块和我们的数组的大小相同的内存。

为什么我们要动态开辟一块内存呢?因为如果我们直接在我们的原数组上面进行修改的话会比较麻烦,开辟新的动态内存也很方便。

然后我们来分析我们的代码,我们开辟完内存以后,我们把数组和我们数组的首尾下标和动态开辟的数组传入到我们的函数里面。

进入到我们的函数内部,我们的这个函数是递归版本的,我们进入到函数以后我们先判断一下我们传过来的数组个数,如果数组个数只有一个的时候,我们就返回,数组个数为1个,这个数组就是有序的,然后我们求出mid中间值,再次给他分组,,,当我们分组分到最后的时候,

我们的函数,进入到10的的时候我们判断,一个数据,返回,然后进入到6去,也是一个数据,返回,最后回到上面,这时候我们就把这两个有序的数组进行合并,合并成一个有序的数据,就变成了下面的数据,那么这个数组也是有序的,然后旁边的也是按这样的方式,合成小的有序的数据,然后这又和我们的这个数据合并,成了大的有序的数据。

所以,我们不管什么时候合数据,都是已经有序的数据,因为递归函数已经把他弄好了。

然后我们来合并这个两个有序的数组,我们把小的放到我们的tmp的前面,大的放到后面,走到最后的话,我们的两个数组里面的数据如果没有排完的话,我们就分别看他们要不要排。

然后我们把我们排好在tmp的数据导入到我们的arr里面。(我们这里是排完一次就导入一次)

代码实现完了,我们来看一下归并排序的时间复杂度:n logn

空间复杂度的话:n,(因为我们额外动态开辟了一个数组空间);

我们来看这个图片,我们的比较算法>排序算法的最终比较:这里我们可以看到我们的归并排序的速度也不慢,图里面被标出的:堆排序,快速排序,归并排序 这三种算法>排序算法的时间复杂度都是n logn

这三种也是比较快的,当然,希尔排序也比较快,希尔排序的时间复杂度为n^1.3,也非常快。

接着就是直接插入排序,然后是直接选择排序,然后是冒泡排序。

好了,前面的7种比较排序我们已经看完了,我们现在来看非比较排序

非比较排序:

计数排序:

计数排序⼜称为鸽巢原理,是对哈希直接定址法的变形应⽤。

操作步骤:

1)统计相同元素出现次数

2)根据统计的结果将序列回收到原来的序列中

我们看这个图片,这个就是我们的计数排序。

我们先创建一个数组,让数组里面的数据全部为0。

当我们传过来数组的时候,我们先遍历我们的数组,当我们开始走到6的时候,我们就让下标为6的数据加1,然后再走到1,我们就让下标为1的数据++,然后我们走到了2,我们就让下标为2的数据++;然后继续。。。。直到最后我们把数组遍历完了,然后我们创造的数组,里面就会变成我们图中的样子,我们有了这个数组以后,我们就可以直接对我们的数据进行排序了,我们就看什么数据有几个,从前往后的排列,就比如我们图里面的数组,我们看到数据1有两个,我们就在我们的数组的前两个位置都放上数据1,然后数据2也是有两个,我们就放两个数据2。。。。最后我们就能排好我们的数组。

那么我们该怎么创造我们的数组呢?

跟上面的一样取出数组里面的最大值然后+1,创造一个这么大的数组吗?

我们来看下面的情况:

我们来看这个图片,这次我们要排的数据是100到109,这时候我们能取他的最大值然后 +1吗?

我们要创造110个空间,但是我们的数组里面最小的数据为100,这就会导致前面下标(0--99)100个空间被浪费掉,这样的话损耗是比较大的,那我们怎么办呢?

我们就在数组里面找到最大的数据,再找到最小的数据,最小的数据,所以这时候我们的数组的空间的大小就是最大的数据 - 最小的数据 + 1。

我们来看这个图,这时候我们的数组的range就是10,然后我们的下标还是0开始的,减去了我们的最小值,

这个就是我们的代码的实现;

当然,当我们要排序的数组是前后大小相差比较大的话,

这时候我们的计数排序可能就不太适合了。

然后我们的这个排序的复杂度:

时间复杂复杂度因为我们的循环,空间复杂度的话,因为我们额外申请了一个range大小的内存空间。

算法>排序算法稳定性的分析:

假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的 相对次序保持不变,即在原序列中,r[i]=r[j],且r[i]在r[j]之前,⽽在排序后的序列中,r[i]仍在r[j]之 前,则称这种算法>排序算法是稳定的;否则称为不稳定的。

什么意思呢?

比如我们的这个数组,当我们排序完后,我们的前面的5还是在后面5的前面的。这样的话这个排序就hi是比较稳定的。

那么,,,

数据结构初阶----完结。


http://www.niftyadmin.cn/n/5869702.html

相关文章

C语言基础要素(006):转义字符入门

转义字符入门 转义字符,顾名思议就是转换字符的意义;一个转义字符在书写上是两个或多个字符,但只表示一个含义。‘\n’就是一个转义字符,当printf函数碰到它时,并没有直接输出字符’\‘与’n’,而是将它们…

SQL命令详解之操作数据库

操作数据库 SQL是用于管理和操作关系型数据库的标准语言。数据库操作是SQL的核心功能之一,主要用于创建、修改和删除数据库对象,如数据库、表、视图和索引等。以下是SQL中常见的数据库操作命令及其功能简介: 1. 查询数据库 查询所有的数据库…

X64 TF位和Single-step单步调试的研究

如果在执行指令时,处理器检测到 EFLAGS 寄存器中的 TF 标志被设置,则会生成单步调试异常。该异常属于陷阱类异常,因为异常是在指令执行之后生成的。处理器不会在设置 TF 标志的指令之后立即生成此异常。例如,如果使用 POPF 指令设…

一周学会Flask3 Python Web开发-Jinja2模板继承和include标签使用

锋哥原创的Flask3 Python Web开发 Flask3视频教程: 2025版 Flask3 Python web开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili 不管是开发网站还是后台管理系统,我们页面里多多少少有公共的模块。比如博客网站,就有公共的头部&…

策略模式环境类的实现方式对比

文章目录 1、策略模式2、聚合策略类实现方式一3、聚合策略类实现方式二4、对比5、补充:ApplicationContextAware接口 1、策略模式 近期工作中,需要处理4.x和5.x两个版本的数据,所以自然想到的是策略模式,写一个抽象类&#xff0c…

DeepSeek-R1的一些影响

DeepSeek-R1火爆全球,肯定不仅仅是开源了一篇论文、一个模型那么简单,更多的是其带来的一些影响,这里简单聊聊。 跳出大模型固有开发范式 NLP&大模型技术发展历程大致如下: 规则方法统计学习方法深度学习方法深度学习预训练…

智慧城市招标进入“资质严审期”!谁主数字孪生技术话语权?

2025年1月,《全国统一大市场建设指引》明确要求招标项目需杜绝“逐利性执法”,技术资质审查更趋透明化。与此同时,住建部《危险性较大的分部分项工程专项施工方案严重缺陷清单》将数字孪生平台的数据治理能力纳入工程验收标准。在此背景下&am…

记录Liunx安装Jenkins时的Package ‘jenkins‘ has no installation candidate

1、确保是否安装了Java,如果没有,可通过以下命令进行安装: sudo apt update sudo apt install openjdk-21-jre2、安装Jenkins sudo apt update sudo apt install jenkins执行sudo apt install jenkins时,可能会出现 意思是&…