欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 会展 > 数据结构-排序

数据结构-排序

2025/1/17 4:44:17 来源:https://blog.csdn.net/2302_79031646/article/details/145082482  浏览:    关键词:数据结构-排序

本节我们来简单介绍几种常用到的排序方式.

目录

    • 1. 都有啥常见的排序方式?
    • 2. 直接插入排序(插入排序)
      • 2.1 基本介绍
      • 2.2 图解过程
      • 2.3 参考代码
    • 3. 冒泡排序
      • 3.1 基本介绍
      • 3.2 图解过程
      • 3.3 参考代码
      • 3.4 冒泡 与 直接插入 的比较
    • 4. 希尔排序
      • 4.1 基本介绍
      • 4.2 图解过程
      • 4.3 希尔排序的时间复杂度
      • 4.4 参考代码
    • 5. 选择排序
      • 5.1 基本介绍
      • 5.2 图解过程
      • 5.3 时间复杂度
      • 5.4 参考代码
    • 6. 堆排序
      • 6.1 基本介绍
      • 6.2 参考代码
      • 6.3 图解过程
    • 7. 快速排序
      • 7.1 基本介绍
      • 7.2 单趟快排
      • 7.3 快排的整体逻辑
      • 7.4 霍尔版本快排参考代码
      • 7.5 优化: 三数取中
      • 7.6 优化: 小区间优化
      • 7.7 挖坑法版本
      • 7.8 挖坑法参考代码
      • 7.9 指针版本快排
      • 7.10 指针版本参考代码
      • 7.11 快排的非递归
      • 7.12 非递归参考代码
    • 8. 归并排序
      • 8.1 基本介绍
      • 8.2 归并图解
      • 8.3 具体代码实现
      • 8.4 非递归版本
      • 8.5 拓展话题: 内/外排? 比较/非比较排?
    • 9. 计数排序
      • 9.1 基本介绍
      • 9.2 图解过程
      • 参考代码

1. 都有啥常见的排序方式?

主要的排序方式有下面几种:

  1. 插入排序
  2. 希尔排序
  3. 选择排序
  4. 堆排序
  5. 快速排序
  6. 冒泡排序
  7. 归并排序
  8. 计数排序

实际上, 排序的分类有多种多样, 你可以按照关键特点即下面方式分类:
在这里插入图片描述
你也可以按照内外排序来进行分类, 或者按照是否用到交换原理来进行分类.

2. 直接插入排序(插入排序)

2.1 基本介绍

最基本的排序之一: 直接插入排序.
基本思想: 把一个待排数据插入到已经有序的序列当中. 即[0, end], end+1插入
编码技巧: 先控制单趟, 再控制整体

2.2 图解过程

下面是一个具体的实例过程:
在这里插入图片描述
在这里插入图片描述
时间复杂度是: O(N*N)

2.3 参考代码

在这里插入图片描述

3. 冒泡排序

3.1 基本介绍

这是一个非常简单的排序, 我不再多说.
基本思想: 每次把最大/最小的数冒到最后, 之后再把次大/次小的数据冒泡到倒数第二的位置.
编码技巧: 先控制单趟, 再控制整体.

3.2 图解过程

在这里插入图片描述

时间复杂度是: O(N*N)

3.3 参考代码

在这里插入图片描述

3.4 冒泡 与 直接插入 的比较

在这里插入图片描述

4. 希尔排序

4.1 基本介绍

基本思想:

  1. 预排序 --> 达到一个接近有序的目的
  2. 直接插入排序 --> 达到一个有序的目的

在这里插入图片描述

4.2 图解过程

在这里插入图片描述

4.3 希尔排序的时间复杂度

在这里插入图片描述

4.4 参考代码

在这里插入图片描述

5. 选择排序

5.1 基本介绍

基本思想: 选择最小的数放到第一个位置, 选择第二小的数放到第二个位置
一次选出最大/最小的, 分别放到最左或者最右的~
注意: 选的是下标, 而不是数, 直接根据下标交换即可.

5.2 图解过程

在这里插入图片描述
这个地方有个隐藏大坑: 就是你交换完最小值时, 恰好最大值也刚好是你交换完最小值的位置.

在这里插入图片描述

5.3 时间复杂度

O(N * N)
最好情况下是多少? O(N * N)

理论上, 这个排序比冒泡排序的适应性还差.
在实际情况中, 选择排序比冒泡排序还快一些, 这是因为选择排序虽然最好情况都是O(N * N),
但是对于选择排序是: N + N-2 + N-4 + …
而冒泡是标准的 N + N-1 + N-2 + N-3 + …
冒泡排序的最好情况很难去构成.

5.4 参考代码

在这里插入图片描述

6. 堆排序

6.1 基本介绍

基本思想: 利用堆的特殊结构 + 向下调整算法实现排序.

  1. 在原数组上直接建堆,推荐用向下调整算法,时间复杂度O(N),建大堆
  2. 首尾交换
  3. 再对arr[0]数据进行向下调整
  4. 再重复上述过程

6.2 参考代码

在这里插入图片描述

6.3 图解过程

在这里插入图片描述

7. 快速排序

7.1 基本介绍

基本思想: 选取一个key, 经过处理, 可以使得key左边比key小, key右边比key大. 之后, 对于左右区间不断递归, 知道全部有序.

因为这个排序比较复杂, 我们先来介绍快排的单趟逻辑是如何的?

7.2 单趟快排

在这里插入图片描述

这时候有同学可以就会产生疑问了: 为啥left和right的相遇点一定比key小呢?
答: 因为, 右边先走.
我们还是严谨一点, 进行分类讨论. 对于left和right相遇, 实际上无非就是L遇R, 或者R遇L.
如果是L遇R: R已经指向比key小的值了, 自然相遇位置比key小咯
如果是R遇L:如果L没有动过, 此时L就是key, 然后key自己跟自己交换, 没啥问题
如果L动过, 那说明L和R进行过交换, 进行过交换, 那么此时L的数是之前R换过来的,
也是比key小的数, 所以说呢, 到最后也一定是比key小的数.

7.3 快排的整体逻辑

上面我们仅仅说了快排的单趟逻辑, 下面图解一下整体逻辑:
在这里插入图片描述
细节控制: 在一些特殊情况下, left 必须 left = keyi, 而不是keyi + 1

在这里插入图片描述

7.4 霍尔版本快排参考代码

在这里插入图片描述

7.5 优化: 三数取中

在某些比较特殊的情况下, 快排容易栈溢出崩溃.
为了解决这个问题 -> 三数取中 或者 随机数选key.

问: 为啥有序情况下, 快排甚至比插入都慢?
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

7.6 优化: 小区间优化

在这里插入图片描述

7.7 挖坑法版本

对于使得key左边比key小, 右边比key大这个要求, 我们可以有不同的实现方式, 因此快排有不同的版本.

在这里插入图片描述

7.8 挖坑法参考代码

在这里插入图片描述

7.9 指针版本快排

在这里插入图片描述

7.10 指针版本参考代码

在这里插入图片描述

7.11 快排的非递归

递归写法好处是比较方便, 但是坏处就是容易栈溢出. 我们下面来实现一下快排的非递归.
在这里插入图片描述

7.12 非递归参考代码

在这里插入图片描述

8. 归并排序

8.1 基本介绍

基本思想: 对于一段数据, 我先让他左边一半有序和右边一半有序, 那不就整体有序了?
对于左边这半部分我继续划分为左半部分和右半部分… 右边那半部分同样分割.
如此进行类推分割, 直到一个数据. 一个数据就算是有序了.
然后两两合并, 合并的时候弄成有序即可.

8.2 归并图解

在这里插入图片描述

8.3 具体代码实现

在这里插入图片描述

8.4 非递归版本

在这里插入图片描述

8.5 拓展话题: 内/外排? 比较/非比较排?

在这里插入图片描述

9. 计数排序

9.1 基本介绍

基本思想: 与前面的排序方法不同, 利用数据的出现频率来排序.
特点: 效率很高, 达到了O(N + Range), Range: 频率哈希的数组大小
局限性:

  1. 不适合分散的数据, 更适合相对集中的数据
  2. 不适合浮点数, 字符串, 结构体的数据排序, 只适合整数

9.2 图解过程

在这里插入图片描述

参考代码

在这里插入图片描述


EOF.

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com