总算进入我们的排序相关算法的学习了。相信不管是系统学习过的还是没有系统学习过算法的朋友都会听说过许多非常出名的排序算法,当然,我们今天入门的内容并不是直接先从最常见的那个算法说起,而是按照一定的规则一个一个的介绍。

首先,我们要介绍的排序算法是插入类型的排序算法。顾名思义,插入排序就是将无序的一个或几个记录“插入”到有序的序列中,比较典型的例子就是简单插入排序和希尔排序。

简单插入排序

简单插入排序,也可以叫做直接插入排序。还是先看代码,再来进行下一步的解释。

function InsertSort($arr)
{
$n = count($arr);
for ($i = 1; $i < $n; $i++) { // 开始循环,从第二个元素开始,下标为 1 的
$tmp = $arr[$i]; // 取出未排序序列第一个元素
for ($j = $i; $j > 0 && $arr[$j - 1] > $tmp; $j--) { // 判断从当前下标开始向前判断,如果前一个比当前元素大
$arr[$j] = $arr[$j - 1]; // 依次移动元素
}
// 将元素放到合适的位置
$arr[$j] = $tmp;
}
echo implode(', ', $arr), PHP_EOL;
} InsertSort($numbers); // 49, 38, 65, 97, 76, 13, 27, 49
// 13, 27, 38, 49, 49, 65, 76, 97

代码量不多吧,其实也非常好理解。我们就拿测试数据的前两个数来简单地说明一下。

首先,第一个循环是从 1 开始的,也就是说,第一个取出的未排序序列元素是 tmp = arr[1] ,也就是当前的 tmp = 38 。

然后开始循环,当前的循环是判断 j-1 的元素是否比当前这个 tmp 元素大,如果是的话,进入循环体,arr[1] = arr[0] 。到目前为止,arr[0] 和 arr[1] 现在都是 49 。整个序列是 49,49,65,……

最后让 arr[0] = $tmp ,也就是等于 38 。(循环的时候 j-- 了)。整个序列是 38,49,65,……

通过下面这张图片,我们可以更清楚地看明白整个序列完成排序的过程。

从上面的步骤可以看出,简单插入排序就是从一边开始,先让前面的数据逐步有序的过程。从代码中就可以看出,它是不断地内部地循环中进行 j 的递减,与前面有序的数列进行比对,当发现了自己合适的位置之后,就将数据放到这个位置上。

从代码和我们的分析来看,简单插入排序的时间复杂度是 O(n2) 。同时,它是属于稳定的排序,什么叫稳定排序呢?细心的同学应该发现了,在我们的测试代码中,有两个相同的数据,也就是 49 。稳定的意思就是相同的数据在排序前后的位置不会发生改变,前面的 49 依然是在后面的 49 前面。这就是排序的稳定性。

另外,简单插入排序比较适合初始记录基本有序的情况,当初始记录无序,n 较大时,这个算法的时间复杂度会比较高,不太适合采用。

希尔排序

简单插入排序很好理解吧,希尔排序又是什么鬼呢?别着急,从这个名字我们是看不出什么端倪的,因为这个排序的名字是以它的发现者命名的。实际上,希尔排序还是一个插入排序的算法。

上文中说过,简单插入排序适合基本有序的情况,而希尔排序就是为了提升简单插入排序的效率而出现的,它主要目的是减少排序的 n 的大小以及通过几次排序就让数据形成基本有序的格式。

对于这个算法,我们不能先上代码了,先来看图吧。

看明白了吗?我们其实是将数据进行分组了,每次分组是以一定的增量为基础的,比如我们这个示意图中就是第一次以 5 为增量进行排序,第二次是以 3 为增量。这样第三次排序的时候,增量为 1 ,也就成为一个普通的简单插入排序了。一会我们代码中就会体现出来。

还是按增量为迭代次序进行这三趟排序的具体分析吧:

1)第一次迭代的时候,我们将分组增量设置为 5 ,这时分别有三组数据,也就是 49 和 13,38 和 27,65 和 49 ,然后对这三组数据进行简单插入排序,之后的数组结果是 13、27、49、97、76、49、38、65 。

2)第二次迭代,分组增量为 3,这时就分成了两组,每组三个数据,分别是 13、97、38 为一组,另一组是 27 、76 、65 。对这两组数据进行简单插入排序之后更新数组结果为 13、27、49、38、65、49、97、76 。

3)其实从两次分组排序之后就可以看出,这个数组已经基本有序了。这时最后就是以分组增量 1 再次进行简单插入排序。说白了,最后这一步就是一个普通的简单插入排序的过程了。

分步骤讲解之后是不是清楚很多了,再重复一篇,希尔排序其实就是按分组来一次大范围的插入排序,最后一步步缩小到只有 1 次增量的简单插入排序了。我们再来看看代码吧:

function ShellSort($arr)
{
$n = count($arr);
$sedgewick = [5, 3, 1]; // 初始的增量值不能超过待排序列的长度
for ($si = 0; $sedgewick[$si] >= $n; $si++); // 开始分组循环,依次按照 5 、3 、 1 进行分组
for ($d = $sedgewick[$si]; $d > 0; $d = $sedgewick[++$si]) {
// 获取当前的分组数量
for ($p = $d; $p < $n; $p++) {
$tmp = $arr[$p];
// 插入排序开始,在当前组内
for ($i = $p; $i >= $d && $arr[$i - $d] > $tmp; $i -= $d) {
$arr[$i] = $arr[$i - $d];
}
$arr[$i] = $tmp;
}
}
echo implode(', ', $arr), PHP_EOL;
}
ShellSort($numbers);

看着代码貌似有三层 for 循环呀,它哪里提升了效率了呢?其实希尔排序的效率提升确实是有限的,它其实是通过前几次的分组让数据先基本有序。而在分组的状态中,数据比较的数量并不会达到 n 的级别。当最后一次进行简单排序的时候,整个数据已经是基本有序了,在这种情况下交换的次数明显也会减少很多,所以它的时间复杂度在理想状态下可以减少到 O(log2n)2 的水平。

总结

排序的入门餐怎么样?我们可不是直接就拿烂大街的冒泡和快排上手的吧。不出名不意味着不会用到,比如我面试的时候曾经有个公司就是在面试题上写明了不能使用冒泡和快排。这时候,我相信简单插入排序直观好理解的特性一定就能帮助我们度过这种面试难关了哦!

测试代码:

https://github.com/zhangyue0503/Data-structure-and-algorithm/blob/master/7.排序/source/7.1插入类排序:简单插入、希尔排序.php

参考文档:

本文示例选自 《数据结构》第二版,严蔚敏

《数据结构》第二版,陈越

关注公众号:【硬核项目经理】获取最新文章

添加微信/QQ好友:【xiaoyuezigonggong/149844827】免费得PHP、项目管理学习资料

知乎、公众号、抖音、头条搜索【硬核项目经理】

B站ID:482780532

【PHP数据结构】插入类排序:简单插入、希尔排序的更多相关文章

  1. SDUT OJ 3403 数据结构实验之排序六:希尔排序

    数据结构实验之排序六:希尔排序 Time Limit: 1000 ms Memory Limit: 65536 KiB Submit Statistic Discuss Problem Descrip ...

  2. SDUT 3403 数据结构实验之排序六:希尔排序

    数据结构实验之排序六:希尔排序 Time Limit: 1000MS Memory Limit: 65536KB Submit Statistic Problem Description 我们已经学习 ...

  3. SDUT-3403_数据结构实验之排序六:希尔排序

    数据结构实验之排序六:希尔排序 Time Limit: 1000 ms Memory Limit: 65536 KiB Problem Description 我们已经学习了各种排序方法,知道在不同的 ...

  4. 数据结构与算法之PHP排序算法(希尔排序)

    一.基本思想 希尔排序算法是希尔排序,也称递减增量排序算法,是插入排序的一种更高效的改进版本. 该方法的基本思想是:先将整个待排元素序列分割成若干个子序列(由相隔某个“增量”的元素组成的)分别进行直接 ...

  5. 我的Java开发学习之旅------>Java经典排序算法之希尔排序

    一.希尔排序(Shell Sort) 希尔排序(Shell Sort)是一种插入排序算法,因D.L.Shell于1959年提出而得名. Shell排序又称作缩小增量排序. 二.希尔排序的基本思想 希尔 ...

  6. 算法相关——Java排序算法之希尔排序(五)

    个子块,即{3,5},{1,0},{5,2},{9,4},{6,12},将每个子块进行插入排序(即第i位与第i+5位进行比较交换),初步排序结果为{3,0,2,4,6,5,1,5,9,12}.希尔排序 ...

  7. 【DS】排序算法之希尔排序(Shell Sort)

    一.算法思想 希尔排序,也称递减增量排序算法,是插入排序的一种更高效的改进版本.希尔排序是非稳定排序算法.希尔排序是基于插入排序的以下两点性质而提出改进方法的:1)插入排序在对几乎已经排好序的数据操作 ...

  8. 算法与数据结构(十三) 冒泡排序、插入排序、希尔排序、选择排序(Swift3.0版)

    本篇博客中的代码实现依然采用Swift3.0来实现.在前几篇博客连续的介绍了关于查找的相关内容, 大约包括线性数据结构的顺序查找.折半查找.插值查找.Fibonacci查找,还包括数结构的二叉排序树以 ...

  9. 数据结构与算法系列——排序(4)_Shell希尔排序

    1. 工作原理(定义) 希尔排序,也称递减增量排序算法,是插入排序的一种更高效的改进版本.但希尔排序是非稳定排序算法. 希尔排序的基本思想是:先将整个待排序的记录序列分割成为若干子序列分别进行直接插入 ...

  10. 数据结构实验之排序六:希尔排序 (SDUT 3403)

    其实,感觉好像增量不同的冒泡,希尔排序概念以后补上. #include <bits/stdc++.h> using namespace std; int a[10005]; int b[1 ...

随机推荐

  1. srt文件的时间轴平移处理

    有时srt字幕文件与视频文件的时间不完全吻合,有一个时间差,这就需要对srt文件的时间轴进行平移,具备这个功能的软件很多,比如:Subtitle Tool, subresync, sabbu, Sub ...

  2. 解决docker删除加载失败的镜像报错

    背景: 准备在vulhub复现weblogic反序列化漏洞时报错,环境加载失败准备删除weblogic镜像时报错: unable to delete 7d35c6cd3bcd (must be for ...

  3. sqli-labs lesson 38-45

    从page3也就是less 38开始进入了堆叠注入(stacked injection) stacked injection: 简单来说就是进行SQL注入时注入了多条语句.因为之前我们都是只进行过注入 ...

  4. 面对对象4 Mixins机制 内置方法 反射 异常

    Mixins机制 为什么要有:子类继承父类的时候,可能会碰到需要继承多个父类的情况,那么继承关系也分主类和辅类,既保持主类的功能,也有辅类的功能. 命名方式,我们需要将主类和辅类区分开来,python ...

  5. noip42

    T1 朴素dp很好想,设 \(dp_{u,0/1}\) ,表示以 \(u\) 为根的子树,选/不选 \(u\) 所产生的最大贡献. 转移方程则有, \[dp_{u,0} = \prod_{v\in s ...

  6. javascript,html,正则表达式,邮箱密码验证

    <!DOCTYPE html> <html>     <head>         <meta charset="utf-8">   ...

  7. UWP 动画之路径

    xml --------------------------------------------- <Page x:Class="MyApp.MainPage" xmlns= ...

  8. mfc HackerTools释放资源

    作用: 在VC环境中除了我们所常用的Dialog.Menu和Bitmap等标准资源类型之外,它还支持自定义资源类型(Custom Resource),我们自定义的资源类型能做些什么呢?呵呵,用处多多. ...

  9. numpy、pandas、scipy、matplotlib、jieba、 openpyxl、pillow的安装

    cmd环境下进入python安装包里的Script文件夹 安装numpy 安装pandas 安装scipy 安装matplotlib 安装jieba(应该是之前装的库安装依赖时下载了) 安装openp ...

  10. Mac Ideal 常用快捷键

    智能提示 ⌘ -> command ⇧ -> shift ⌥ -> option -> 上箭头 -> 下箭头 ⌃ -> Control mac的option键  = ...