news 2026/4/26 9:54:05

希尔排序--自学笔记

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
希尔排序--自学笔记

希尔排序

学习目标:

一.希尔排序的思想

二.增量序列

三.复杂度分析

四.希尔排序为什么快?

五.强化练习

一.希尔排序的思想

1.将待排序数组按照一定的“间隔”分为多个子数组,

每组分别进行“插入排序”

2.逐渐缩小间隔,进行下一轮排序

3.最后一轮时,取间隔为1,相当于直接使用插入排序

但此时经过 [宏观调控] 数组已经基本有序

所以此时插入排序只需进行少量交换

代码实现
publicstaticvoidshellSort(int[]arr){for(intgap=arr.length/2;gap>0;gap/=2){for(intgroupStartIndex=0;groupStartIndex<gap;groupStartIndex++){for(intcurrentIndex=groupStartIndex+gap;currentIndex<arr.length;currentIndex+=gap){intcurrentNumber=arr[currentIndex];intpreIndex=currentIndex-gap;while(preIndex>=groupStartIndex&&arr[preIndex]>currentNumber){arr[preIndex+gap]=arr[preIndex];preIndex-=gap;}arr[preIndex+gap]=currentNumber;}}}}

目前的思路是,根据gap分组,处理完一组后,调整指针,再处理下一组

这种思路非常符合人类思维,但对于计算机而言,它更喜欢

从gap开始,依次向前,将每个数字插入到其对应的组中,

虽然数字好像在不同组间跳跃,但对于计算机就像是在访问一段连续的数组

代码优化

将第二、三个for循环整合成一个

publicstaticvoidshellSort(int[]arr){for(intgap=arr.length/2;gap>0;gap/=2){for(intcurrentIndex=gap;currentIndex<arr.length;currentIndex++){intcurrentNumber=arr[currentIndex];intpreIndex=currentIndex-gap;while(preIndex>=0&&arr[preIndex]>currentNumber){arr[preIndex+gap]=arr[preIndex];preIndex-=gap;}arr[preIndex+gap]=currentNumber;}}}

二.增量序列

1.定义

Shell排序中,每一遍排序的间隔 ‘gap’ 被称为 “增量” ,所有增量组成的序列叫做增量序列

也就是上述例子中的’5‘,’2‘,’1‘,增量是依次递减

所以Shell排序又称为缩小增量排序

增量序列的选择会极大影响Shell排序的效率!

若没有正确选择增量序列,Shell排序效率可能比插入排序更低!

如图所示,在gap = 8 , 4 , 2 时无元素交换,只有在gap = 1时才发挥了作用

而此时Shell排序相比插入排序,多做了无用功

2.一些著名的增量序列

(1) Hibbard增量序列

Dk= 2k- 1 , 也就是 1,3,7,15,… 。

数学界猜想它最坏的时间复杂度为O(n3/2) ,

平均时间复杂度为O(n5/4)

(2) Knuth增量序列

D1= 1; Dk+1= 3 * Dk+ 1,也就是 1,4,13,40,… 。

数学界猜想它的平均时间复杂度为O(n3/2)

(3) Sedgewick增量序列

1,5,19,41,109,… 。

这个序列的元素有的是通过 9 * 4k- 9 * 2k+ 1 计算出来的

有的是通过 4k- 3 * 2k+ 1 计算出来的

并将其按照从小到大排列

数学界猜想它最坏的时间复杂度为O(n4/3)

平均时间复杂度为O(n7/6)

3.增量序列示例

publicstaticvoidshellSort(int[]arr){//使用Knuth增量序列//先要找出增量序列的最大值intmaxKnuthNum=1;//初始化while(maxknuthNum*3+1<arr.length){maxKnuthNum=maxknuthNum*3+1}//把值赋给gap,并修改gap变化的规律for(intgap=maxknuthNum;gap>0;gap=(gap-1)/3){for(intcurrentIndex=gap;currentIndex<arr.length;currentIndex++){intcurrentNumber=arr[currentIndex];intpreIndex=currentIndex-gap;while(preIndex>=0&&arr[preIndex]>currentNumber){arr[preIndex+gap]=arr[preIndex];preIndex-=gap;}arr[preIndex+gap]=currentNumber;}}}

三.复杂度分析

由于增量序列会极大影响Shell排序的效率

时间复杂度:平均时间复杂度介于O(n)到O(n2)

​ 普遍认为最好的时间复杂度为O(n1.3)

空间复杂度:O(1) 只需要常数级的临时变量

四.希尔排序为什么快?

1.“逆序对”概念引入

当我们从小到大排序时,在数组中的两个数字,如果前面一个数字大于后面的数字

则这两个数字组成一个逆序对

而所有排序算法的本质都是:消除逆序对

对于随机的数组,逆序对的数量是O(n2)级的

如果使用交换相邻元素的方法来消除逆序对,每次只能消除一组逆序对

所以交换的次数是O(n2)级的

这就是冒泡、选择、插入排序算法时间复杂度只能达到O(n2)的原因

基于交换元素的排序算法 ( 空间复杂度为O(1) ) 想突破O(n2),必须通过一些比较,交换间隔比较远的元素

也就是需要在一次交换中能够消除一个以上的逆序对

Shell排序算法就是如此

此后的快排、堆排也是如此…

五.强化练习

506. 相对名次 - 力扣(LeetCode)

杂度为O(1) ) 想突破O(n2),必须通过一些比较,交换间隔比较远的元素

也就是需要在一次交换中能够消除一个以上的逆序对

Shell排序算法就是如此

此后的快排、堆排也是如此…

五.强化练习

506. 相对名次 - 力扣(LeetCode)

912. 排序数组 - 力扣(LeetCode)

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

别再手写过滤器!SpringCloud Gateway 内置30 个,少写 80% 重复代码

发现公司的网关项目里有很多的轮子&#xff0c;几乎每个人接手这个项目开发&#xff0c;都会自定义过滤器&#xff0c;导致有非常非常多的过滤器&#xff0c;修改其中一个&#xff0c;指不定就会影响其他的人功能&#xff0c;非常的恼火。其实在 Spring Cloud Gateway 本身内置…

作者头像 李华
网站建设 2026/4/20 22:13:38

开源 + 国产芯片:具身智能的 DeepSeek 时刻来了?

为什么大语言模型已经能写作、编程、拿数学奥赛金牌&#xff0c;但机器人连叠个衣服都费劲&#xff1f;一个反常识的事实&#xff1a; 机器人全行业的训练数据加起来&#xff0c;也不够特斯拉的自动驾驶系统训练一天的。 这一点不夸张。 特斯拉 FSD&#xff08;Full Self-Drivi…

作者头像 李华
网站建设 2026/4/17 9:38:42

PHP 8.5 新特性 闭包可以作为常量表达式了

默认值以前,我会写这样的代码:function someFunction(mixed $someValue, ?callable $callback null): bool{$callback ?? fn () > true;return $callback($someValue);}或者这样:final class SomeClass{private Closure $someCallable;public function __construct(){$t…

作者头像 李华
网站建设 2026/4/19 21:47:28

分布式抢券系统限流器降级与CAP权衡全攻略

分布式抢券系统限流器降级与CAP权衡全攻略 1. 背景与挑战 在电商抢券、秒杀等场景中,限流器是核心组件,用于避免系统过载和优惠券超发。通常我们依赖 Redis 作为全局分布式限流器的计数中心。 然而,一旦 Redis 集群出现故障或网络分区,系统必须具备 多层次的降级与兜底能…

作者头像 李华
网站建设 2026/4/25 2:23:57

通达信周KDJ副图

{}{周C:("CCI.CCI#WEEK"(14)/10),COLORGREEN,LINETHICK2; IF(周C>周C AND 周C>REF(周C,5),周C,DRAWNULL),COLORRED,LINETHICK3;}; 周DIF:100*"MACD.DIF#WEEK"COLORGREEN; 周DEA:100*"MACD.DEA#WEEK",COLORCYAN,LINETHICK2; 别冲动反弹:周D…

作者头像 李华