news 2026/3/5 8:18:53

希尔排序与堆排序

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
希尔排序与堆排序

希尔排序

介绍

希尔排序由希尔发明,当一个序列的无序程度较低时,那么使用插入排序的效率就会很快,希尔排序的核心思想就是,通过逐步缩小增量(步长)来逐步把一开始很无序的序列慢慢变得有序,最后当增量为1时,就是一次典型的插入排序。

实现

typedef int ElementType; void ShellSort(ElementType A[], int N) { int i,j,Increment; ElementType tmp; for (Increment=N/2; Increment>0; Increment/=2) { for (i=Increment; i<N; i++) { tmp = A[i]; for (j=i; j>=Increment; j-=Increment) { if (tmp < A[j-Increment]) { A[j] = A[j-Increment]; }else { break; } } A[j] = tmp; } } }

假如有一个元素数量为11的乱序序列,那么就可以得到11/2=5,5/2=2,2/1=1,这三个增量。对序列进行增量为5的插入排序,分别获得下标为(5,0),(6,1),(7,2),(8,3),(9,4),(10,5)这六组下标的跳跃步长为5的序列,然后分别对这六组序列进行插入排序,排完序后的整个数列的有序程度就会升高一些,这样就可以提高之后进行直接插入排序的效率了。

接着进行增量为2的插入排序,分别获得下标为(2,0),(3,1),(4,2,0),(5,3,1),(6,4,2,0),(7,5,3,1),(8,6,4,2,0),(9,7,5,3,1),(10,8,6,4,2,0)这九组下标的跳跃步长为2的序列,再分别对这九组序列进行插入排序,排序完后整个序列的有序程度就很高了。看似这一轮的比较次数很多,但其实还有个break语句,其表示若第一次比较不交换时,那就不进行后续的比较了,例如(4,2,0),若下标为4和2的不需要交换,那说明下标为4,2,0的就已经是有序了,不需要再对2,0进行比较了。若下标为4,2的进行交换了,那还得继续比较下标为4和0的元素。

算法中第一个for循环得到增量矩阵,第二个for循环获得各个分序列中的首元素的下标,第三个for循环对各个分序列进行插入排序。

分析

希尔排序是不稳定排序,其会把不同步长的分序列进行排序,当相同元素被分到不同分序列中时,可能会改变它们的相对位置。

不同步长序列的时间复杂度不一样,若为n/2,n/4....1的步长的话,那最坏的时间复杂度为O().若用Hibbard步长序列(-1),那时间复杂度为O().若用Sedgewick步长序列(混合序列9×-9×+1和-3×+1),那最坏的情况下时间复杂度被优化成O()了。

堆排序

介绍

在之前的文章中介绍了堆这一数据结构,得益于堆的特性,其根节点始终是最大值(或最小值),我们可以不停的取堆的根节点元素然后按顺序排好,这样就可以得到排好序的序列了。

实现

typedef int ElementType; #define LeftChild(i) (2*(i)+1) void Swap(ElementType *a, ElementType *b) { ElementType temp = *a; *a = *b; *b = temp; } void PercDown(ElementType A[],int i,int N) { int Child; ElementType Tmp; for (Tmp = A[i];LeftChild(i)<N;i=Child) { Child = LeftChild(i); if (Child!=N-1 && A[Child+1]>A[Child]) { Child++; } if (Tmp<A[Child]) { A[i] = A[Child]; }else { break; } } A[i] = Tmp; } void HeapSort(ElementType A[],int N) { int i; for (i=N/2;i>=0;i--) { PercDown(A,i,N); } for (i=N-1;i>0;i--) { Swap(&A[0],&A[i]); PercDown(A,0,i); } }

得益于堆对应完全二叉树父子节点位置的数学关系,我们可以通过对原始无序序列进行下滤操作从而将其转化成符合堆结构的数组。

此时这个数组的第一个元素是整个序列的最大值,把它和最后的元素进行交换,这样整个序列中,我们完成了最大元素的归位。然后在把除最后元素的其他所有元素进行下滤操作重新形成新堆,再把第一个元素放到倒数第二个位置上,这样就完成了第二个最大元素的归位,循环往复就以从大到小的顺序完成了排序操作。

分析

堆排序是不稳定的,当把栈顶元素和最后有个元素进行交换时,可能会改变相同元素的相对位置。

要进行n-1次交换,每次交换都得重新通过下滤操作构建堆结构(该构建操作的时间复杂度为 O(LogN)),所以一共是O(NLogN)。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/3 12:25:04

43、有限域算法与确定性素性测试

有限域算法与确定性素性测试 1. 多项式因式分解相关内容 在有限域上进行多项式因式分解是一个重要的研究领域,涉及到多个算法和相关练习,以提升分解效率。 1.1 分离集与多项式因式分解 给定特定条件,集合 $S := {rep(\alpha_i) : 0 \leq i \leq k - 1}$ 是多项式 $g$ 在…

作者头像 李华
网站建设 2026/3/3 0:09:20

python三元赋予我的单位换算器以智能(表达式函数展示)

#算法#自研工具#代码艺术#抒写范式#三赢代码 注&#xff1a;此文10-day后将收入专栏我的思想自研工具 三元赋予涨灵智&#xff0c;脱模成型生景致。 笔记模板由python脚本于2025-12-14 23:08:50创建&#xff0c;本篇笔记适合喜欢考究代码的coder翻阅。 学习的细节是欢悦的历程 …

作者头像 李华
网站建设 2026/3/4 8:00:56

深蓝词库转换:轻松实现20+输入法词库互转的终极指南

深蓝词库转换&#xff1a;轻松实现20输入法词库互转的终极指南 【免费下载链接】imewlconverter ”深蓝词库转换“ 一款开源免费的输入法词库转换程序 项目地址: https://gitcode.com/gh_mirrors/im/imewlconverter 还在为不同输入法间的词库不兼容而烦恼吗&#xff1f;…

作者头像 李华
网站建设 2026/2/26 22:48:05

Java Excel处理性能革命:FastExcel实现20倍加速的终极方案

Java Excel处理性能革命&#xff1a;FastExcel实现20倍加速的终极方案 【免费下载链接】fastexcel Generate and read big Excel files quickly 项目地址: https://gitcode.com/gh_mirrors/fas/fastexcel 在当今数据驱动的时代&#xff0c;Excel文件处理已成为Java开发中…

作者头像 李华
网站建设 2026/3/4 16:07:51

ELK+Filebeat实战

文章目录 前言一、什么是ELK二、ELK核心组件说明1、Elasticsearch1.1、什么是Elasticsearch1.2、Elasticsearch 作用1.3、Elasticsearch 应用场景1.4、Elasticsearch 工作原理 2、Logstash2.1、什么是Logstash2.2、Logstash作用2.3、Logstash应用场景2.4、Logstash工作原理 3、…

作者头像 李华
网站建设 2026/2/11 19:54:57

Lan Mouse终极指南:如何实现多设备鼠标键盘无缝共享?

Lan Mouse终极指南&#xff1a;如何实现多设备鼠标键盘无缝共享&#xff1f; 【免费下载链接】lan-mouse mouse & keyboard sharing via LAN 项目地址: https://gitcode.com/gh_mirrors/la/lan-mouse 在日常工作中&#xff0c;你是否经常需要在多台电脑之间来回切换…

作者头像 李华