希尔排序(C++实现)

文章目录

  • 前言
  • 1. 基础概念
  • 2. 动图演示
  • 3. 代码实现
  • 4. 排序过程
  • 5. 效率分析
  • 6. 总结


前言

上篇文章讲了直接插入排序算法。

首先,在待排序的数组中,元素本身就是有序的情况下,就不需要移动任何元素,所以直接插入排序最好情况时间复杂度为 O ( n ) O(n) O(n)。但是,如果数组中元素多数都是有序(基本有序)的情况下,那么需要移动的元素数量就会大大减少。

这里所谈到的基本有序,可以理解成小的关键字大部分在前面,大的关键字大部分在后面,比如如下数组中的元素 {1,2,10,16,18,45,23,99,42,67}{16,1,45,23,99,2,18,67,42,10} 数组中元素相比,前者算得上是基本有序了。

其次,如果数组中元素数量较少,那么直接插入排序的效率也会很高。基于上述两点对直接插入排序算法进行改进,就可以得到希尔排序算法。

1. 基础概念

希尔排序这个名字来源于它的发明者希尔(Donald Shell),这类排序也被称作 “缩小增量排序(Diminishing Increment Sort)”,是直接插入排序的一种更高效率的改进版。

希尔排序算法的思想是先追求元素的部分有序,然后再逐渐逼近全局有序。具体的做法是先将整个待排序记录序列(或称为数组元素)分割成若干个子序列,分别进行直接插入排序,等到整个序列中的记录基本有序时,再对所有记录进行一次直接插入排序。

希尔排序会进行多趟排序,每趟排序会设置一个增量,这里注意,这个增量初始值到底是多大才合适并没有公论,比如可以将 “数组中元素数量 / 2" 作为增量的初始值。

这里以数组 {1,2,10,16,18,45,23,99,42,67} 为例,来说明希尔排序。

(1)首先,数组中的元素有 10 个,所以增量初值可以设置为 5。第一趟排序,把距离为 5 的元素划分到一个子序列中,并对这个子序列中的元素从小到大排序。如下图所示:

在这里插入图片描述

从上图可以看到,第一趟排序,因为增量值是 5,这意味着即将排序的元素间隔 5 个位置。所以下标为 0 的元素 16 和下标为 5 的元素 2 需要进行大小比较,并根据从小到大的顺序决定谁在前谁在后。因为 2 比 16 小,所以 2 应该在前,也就是 16 和 2 互换位置。接着,有元素 1 和 18、元素 45 和 67、元素 23 和 42、元素 99 和 10 依次进行大小比较并排序,本趟排序得到的结果数组元素值为 {2,1,45,23,10,16,18,67,42,99}

(2)第二趟排序,要缩小增量的值,比如可以每次缩小一半(希尔本人这样建议),也就是:增量 = 增量 / 2,原来增量的值是 5,这次增量的值就变成了 2,即把距离为 2 的元素划分到一个子序列中并对该子序列中的元素从小到大排序。如下图所示:

在这里插入图片描述

从上图可以看到,第二趟排序因为增量值是 2,意味着即将排序的元素间隔 2 个位置。所以下标为 0 的元素 2、下标为 2 的元素 45、下标为 4 的元素 10、下标为 6 的元素 18、下标为 8 的元素 42 进行大小比较并根据从小到大的顺序决定谁在前谁在后。最终得到 {2、10、18、42、45} 的顺序。

同理,元素 {1、23、16、67、99} 从小到大排序,本趟排序得到的结果数组元素值为 {2,1,10,16,18,23,42,67,45,99}。可以看到,每一趟排序,都使数组中的元素更进一步基本有序。

(3)第三趟排序,继续缩小增量的值,增量 = 增量 / 2,原来增量的值是 2,这次增量的值就变成了 1。这就意味着要对数组中的所有元素进行从小到大的直接插入排序,增量值为 1 后的排序也是最后一次排序。最终数组元素的值就变成了 {1,2,10,16,18,23,42,45,67,99}

所以,一共进行了三趟排序,得到了最终排好序的数组元素。如下图所示:

在这里插入图片描述

2. 动图演示

可以看下面的动图演示结果:

在这里插入图片描述

3. 代码实现

代码如下:

#include <iostream>
using namespace std;template<typename T>
void ShellSort(T arr[], int len)
{if (len <= 1) //不超过1个元素的数组,没必要排序return; int gap = len / 2; //gap: 增量,取值分别为7、3、1while (gap >= 1){//每一趟采用直接插入排序来实现(实现代码与直接插入排序其实是一摸一样) //第一次while循环:i=7~14;第二次while循环:i=3~14;第三次while循环i=1~14;for (int i = gap; i < len; ++i) //i值每次改变可以处理到不同的子序列{if (arr[i] < arr[i - gap]){T temp = arr[i]; ;//暂存arr[i]值,防止后续移动元素时值被覆盖int j;for (j = i - gap; j >= 0 && arr[j] > temp; j -= gap) //检查所有前面排好序的元素{arr[j + gap] = arr[j]; //所有大于temp的元素都向后移动}arr[j + gap] = temp; //复制数据到插入位置,注意j因为被减了gap,这里加回来}}cout << "本趟希尔排序,增量为: " << gap <<" "<<"结果为: ";for (int i = 0; i < len; ++i)cout << arr[i] << " ";cout << endl;gap /= 2; //增量每次减半	 }return;
}

在主函数中,加入一个包含 15 个元素的数组,然后进行从小到大排序。

int main()
{int arr[] = { 67, 1, 45, 23, 99, 2, 18, 16, 42, 10, 8, 44, 106, 29, 4};int len = sizeof(arr) / sizeof(arr[0]);cout <<"原始数据为:"; for (int i = 0; i < len; ++i) cout << arr[i] << " ";cout << endl;ShellSort(arr, len); cout << "希尔排序后的数据为:"; for (int i = 0; i < len; ++i) cout << arr[i] << " ";cout << endl;return 0; //我低洼地 
} 

排序结果如下:

在这里插入图片描述

从结果可以看到,一共进行了三趟排序,增量值分别为 7、3、1。

4. 排序过程

希尔排序算法的代码实现并不复杂,但可能理解起来不那么直观,这里阐述一下程序的执行步骤。

当增量值为 7 时,程序是怎么工作的呢?如下图所示:

在这里插入图片描述

上图展示的是第一趟排序(增量为 7)的程序执行流程,为了清晰,分成了几个子图来绘制。最上面是待排序的原始数组数据。排序时先看子图 1:

  • 67 和 16 交换位置。
  • 1 和 42 不需要交换位置。
  • 45 和 10 交换位置。
  • 23 和 8 交换位置。
  • 99 和 44 交换位置。
  • 2 和 106 不需要交换位置。
  • 18 和 29 不需要交换位置。

这里值得说的是元素 4:

  • 元素 4 先和 67 交换位置,如子图 2。
  • 元素 16 再和 4 交换位置,如子图 3。
  • 子图 2 和子图 3 这两次连续的交换位置是通过代码中最内层 for 循环实现的。

所以,经过第一趟排序后,数组中的元素内容就是:{4, 1, 10, 8, 44, 2, 18, 16, 42, 45, 23, 99, 106, 29, 67}

接着,增量值减少为原来的一半,从 7 变为了 3,开始第二趟排序,增量值为 3 时,程序是怎么工作的呢?如下图所示:

在这里插入图片描述

其中:

  • 4 和 8 不需要交换位置。
  • 1 和 44 不需要交换位置。
  • 10 和 2 交换位置。
  • 8 和 18 不需要交换位置。
  • 44 和 16 交换位置。
  • 10 和 42 不需要交换位置。
  • 18 和 45 不需要交换位置。
  • 44 和 23 交换位置,如子图 2。
  • 42 和 99 不需要交换位置。
  • 45 和 106 不需要交换位置。
  • 44 和 29 交换位置,如子图 3。
  • 99 和 67 交换位置。

所以,经过第二趟排序后,数组中的元素内容就是:{4, 1, 2, 8, 16, 10, 18, 23, 42, 45}

接着,增量值减少为原来的一半,从 3 变为了 1,开始第三趟排序,此时,程序的执行步骤与前面讲述的直接插入排序步骤完全一致,这里就不再赘述,经过了三趟排序,数组中的元素已经按照从小到的顺序排好了,数组中元素的最终内容就是:{1, 2, 4, 8, 10, 16, 18, 23, 29, 42, 41, 45, 67, 99, 106}

5. 效率分析

从代码中可以看到,希尔排序实现代码的空间复杂度为 O ( 1 ) O(1) O(1)。而对于时间复杂度的分析,本算法则显得比较复杂。当采用不同的增量序列,比如上面代码中每次增量减少为原来的一半时,希尔排序的总趟数会不同,而且每趟排序元素对比次数和元素移动次数都可能会受到影响。所以希尔排序的时间复杂度目前为止还无法用数学手段确切地证明。

但如果增量的初值直接设置为 1 的话,那么希尔排序会退化为直接插入排序,这时的时间复杂度是希尔排序的最坏时间复杂度即 O ( n 2 ) O(n^2) O(n2)。如果待排序元素的数量在一定的范围内,那么时间复杂度可以达到 O ( n 1.3 ) O(n^{1.3}) O(n1.3),平均时间复杂度为 O ( n l o g 2 n ​ ) O(nlog^n_2​) O(nlog2n),这意味着希尔排序算法优于直接插入排序算法。

此外,希尔排序算法是不稳定的,这不难证明。试想一下具有 3 个元素 99、10、10 的数组,因为增量值的设定并没有公论,所以如果设定第一趟排序增量值为 2,第二趟排序增量值为 1,那么后面两个都是 10 的数组元素位置就会发生改变。如下图所示:

在这里插入图片描述

上图所示,第一趟排序元素 99 和最末尾的元素 10 进行了位置交换,而第二趟排序并没有做任何数组元素位置的交换。但显而易见,原下标为 2 的数组元素 10 被移动到了下标为 0 的位置,跑到了下标为 1 的数组元素 10 之前。所以,希尔排序算法是不稳定的。

6. 总结

希尔排序算法是对直接插入排序算法的改进,它先追求元素的部分有序,然后再逐渐逼近全局有序。

希尔排序通过进行多趟排序的方式来达成,排序开始时会选择一个增量,根据增量将待排序序列分成若干个子序列,对每个子序列进行直接插入排序。然后逐步缩小增量并继续根据增量将待排序序列分成若干个子序列并对每个子序列进行直接插入排序,直到增量变为了 1 才完成了整个希尔排序的过程。

与直接插入排序相比,希尔排序的速度更快。同时,通过对增量的调整也许能进一步加速排序效率。不过,希尔排序的实现代码更加复杂,且选择合适的增量也显得特别重要。此外还要注意,希尔排序算法是不稳定的。

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

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

相关文章

使用win_b64做CATIA的开发测试

文章目录 一、把win_b64文件夹放在无中文的路径下二、启动环境文件编辑器三、新建环境文件四、集成CAA代码五、去用户环境目录查看环境文件是否生成六、关闭环境文件编辑器七、去桌面双击新建的快捷方式即可启动CATIA&#xff0c;进行测试八、进入对应的开发模块&#xff0c;查…

第十二届2023软件杯国家二等奖赛后感想总结

一&#xff0c;相关链接 软件杯官网&#xff1a;软件杯大赛官网 (cnsoftbei.com) 金蝶赛道&#xff1a;金蝶云苍穹开发者门户 (kingdee.com) 二&#xff0c;个人介绍 首先我是个双非院校的学生&#xff0c;专业为计算机科学与技术&#xff0c;打这个比赛是在大二下的暑假开始的…

想要精通算法和SQL的成长之路 - 二叉树的序列化和反序列化问题

想要精通算法和SQL的成长之路 - 二叉树的序列化和反序列化问题 前言一. 二叉树的层序遍历&#xff08;BFS&#xff09;二. 二叉树的序列化与反序列化2.1 序列化操作2.2 反序列化操作 前言 想要精通算法和SQL的成长之路 - 系列导航 一. 二叉树的层序遍历&#xff08;BFS&#xf…

RabbitMQ-主题模式

接上文 RabbitMQ-发布订阅模式和路由模式 1 主题模式 #通配符 代表0个或多个。*通配符 代表 1个或多个 进行测试&#xff0c;修改配置文件 Configuration public class RabbitConfiguration {Bean("topicExchange") //这里使用预置的Topic类型交换机public Exchan…

JMETER自适应高分辨率的显示器

系列文章目录 历史文章 每天15分钟JMeter入门篇&#xff08;一&#xff09;&#xff1a;Hello JMeter 每天15分钟JMeter入门篇&#xff08;二&#xff09;&#xff1a;使用JMeter实现并发测试 每天15分钟JMeter入门篇&#xff08;三&#xff09;&#xff1a;认识JMeter的逻辑控…

nginx下载与安装教程

文章目录 nginx简介nginx的主要应用场景nginx开源项目的源码结构 使用centos7安装nginx检查centos版本号和linux内核版本检查是否安装gcc、pcre、zlib、openssl等依赖 安装nginx启动nginx停止nginx重启nginx nginx简介 nginx是一款业内流行、功能强大的web服务器。 高性能&…

FFmpeg 命令:从入门到精通 | ffplay 播放控制选项

FFmpeg 命令&#xff1a;从入门到精通 | ffplay 播放控制选项 FFmpeg 命令&#xff1a;从入门到精通 | ffplay 播放控制选项选项表格图片 FFmpeg 命令&#xff1a;从入门到精通 | ffplay 播放控制选项 选项表格 项目说明Q&#xff0c;Esc退出播放F&#xff0c;鼠标左键双击全…

[React] React高阶组件(HOC)

文章目录 1.Hoc介绍2.几种包装强化组件的方式2.1 mixin模式2.2 extends继承模式2.3 HOC模式2.4 自定义hooks模式 3.高阶组件产生初衷4.高阶组件使用和编写结构4.1 装饰器模式和函数包裹模式4.2 嵌套HOC 5.两种不同的高阶组件5.1 正向的属性代理5.2 反向的继承 6.如何编写高阶组…

【C语言】函数的定义、传参与调用(二)

&#x1f497;个人主页&#x1f497; ⭐个人专栏——C语言初步学习⭐ &#x1f4ab;点击关注&#x1f929;一起学习C语言&#x1f4af;&#x1f4ab; 目录 导读&#xff1a; 1. 函数的嵌套调用 1.1 什么是嵌套调用 1.2 基础实现 1.3 调用流程解析 2. 函数的链式访问 2.1 …

CCF CSP认证 历年题目自练Day21

题目一 试题编号&#xff1a; 201909-1 试题名称&#xff1a; 小明种苹果 时间限制&#xff1a; 2.0s 内存限制&#xff1a; 512.0MB 题目分析&#xff08;个人理解&#xff09; 先看输入&#xff0c;第一行输入苹果的棵树n和每一次掉的苹果数m还是先如何存的问题&#xf…

小谈设计模式(16)—抽象工厂模式

小谈设计模式&#xff08;16&#xff09;—抽象工厂模式 专栏介绍专栏地址专栏介绍 抽象工厂模式结构抽象工厂&#xff08;AbstractFactory&#xff09;具体工厂&#xff08;ConcreteFactory&#xff09;抽象产品&#xff08;AbstractProduct&#xff09;具体产品&#xff08;C…

2023年10月4日

服务器 #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);//实例化一个服务器server new QTcpServer(this);//此时&#xff0c;服务器已经成功进入监听状态&…

ubuntu22.04 x11窗口环境手势控制

ubuntu22.04 x11窗口环境手势控制 ubuntu x11窗口环境的手势控制并不优秀&#xff0c;我们可以使用touchegg去代替 这个配置过程非常简单&#xff0c;并且可以很容易在一定范围内达到你想到的效果&#xff0c;类比mac的手势控制 关于安装 首先添加源&#xff0c;并安装 sud…

进程调度的时机,切换与过程以及方式

1.进程调度的时机 进程调度&#xff08;低级调度〉&#xff0c;就是按照某种算法从就绪队列中选择一个进程为其分配处理机。 1.需要进行进程调度与切换的情况 1.当前运行的进程主动放弃处理机 进程正常终止运行过程中发生异常而终止进程主动请求阻塞&#xff08;如等待l/O)…

电脑通过串口助手和51单片机串口通讯

今天有时间把电脑和51单片机之间的串口通讯搞定了&#xff0c;电脑发送的串口数据&#xff0c;单片机能够正常接收并显示到oled屏幕上&#xff0c;特此记录一下&#xff0c;防止后面自己忘记了怎么搞得了。 先来两个图片看看结果吧&#xff01; 下面是串口3.c的文件全部内容&a…

OpenGLES:绘制一个混色旋转的3D圆锥

效果展示&#xff1a; 本篇博文总共会实现两种混色旋转的3D圆锥&#xff1a; 一.圆锥解析 1.1 对圆锥的拆解 上一篇博文讲解了绘制圆柱体&#xff0c;这一篇讲解绘制一个彩色旋转的圆锥 在绘制圆柱体时提到过&#xff0c;关键点是先将圆柱进行拆解&#xff0c;便于创建出顶…

计算机网络(五):运输层

参考引用 计算机网络微课堂-湖科大教书匠计算机网络&#xff08;第7版&#xff09;-谢希仁 1. 运输层概述 之前所介绍的计算机网络体系结构中的物理层、数据链路层以及网络层它们共同解决了将主机通过异构网络互联起来所面临的问题&#xff0c;实现了主机到主机的通信&#xff…

Ubuntu安装samba服务器

为了window系统下能够像访问本地目录一样访问ubuntu系统下的目录&#xff0c;这里我通过安装samba服务器&#xff0c;将ubuntu系统的文件目录通过网络挂载的方式共享出来&#xff0c;以便在window下就能够对ubuntu系统的文件进行读写等访问操作&#xff0c;这里记录一下samba服…

ESLint自动修复代码规范错误

基于 vscode 插件 ESLint 高亮错误&#xff0c;并通过配置 自动 帮助我们修复错误 在设置中 settings.json添加这段代码就自动修复错误 // 当保存的时候&#xff0c;eslint自动帮我们修复错误 "editor.codeActionsOnSave": { "source.fixAll": true }, /…

如何使用 AI与人工智能的定义、研究价值、发展阶段

目录 一、什么是人工智能 二、人工智能的研究价值 三、人工智能的发展阶段 一、什么是人工智能 人工智能&#xff08;Artificial Intelligence&#xff0c;简称AI&#xff09;是一门研究如何使计算机能够模拟和执行人类智能活动的科学与技术。人工智能旨在开发智能代理&…