排序算法:希尔排序

1959 年 7 月,美国辛辛那提大学的数学系博士 Donald Shell 在 《ACM 通讯》上发表了希尔排序算法,成为首批将时间复杂度降到 O(n²)以下的算法之一。虽然原始的希尔排序最坏时间复杂度仍然是 O(n²) ,但经过优化的希尔排序可以达到 O(n1.3)甚至O(n7/6)。

略为遗憾的是,所谓「一将功成万骨枯」,希尔排序和冒泡、选择、插入等排序算法一样,逐渐被快速排序所淘汰,但作为承上启下的算法,不可否认的是,希尔排序身上始终闪耀着算法之美。

希尔排序本质上是对插入排序的一种优化,它利用了插入排序的简单,又克服了插入排序每次只交换相邻两个元素的缺点。它的基本思想是:

  • 将待排序数组按照一定的间隔分为多个子数组,每组分别进行插入排序。这里按照间隔分组指的不是取连续的一段数组,而是每跳跃一定间隔取一个值组成一组
  • 逐渐缩小间隔进行下一轮排序
  • 最后一轮时,取间隔为 1,也就相当于直接使用插入排序。但这时经过前面的「宏观调控」,数组已经基本有序了,所以此时的插入排序只需进行少量交换便可完成

举个例子,对数组 [84,83,88,87,61,50,70,60,80,99] 进行希尔排序的过程如下:

第一遍(5 间隔排序):按照间隔 5 分割子数组,共分成五组,分别是

84,50\],\[83,70\],\[88,60\],\[87,80\],\[61,99\]。对它们进行插入排序,排序后它们分别变成: \[50,84\],\[70,83\],\[60,88\],\[80,87\],\[61,99\],此时整个数组变成 \[50,70,60,80,61,84,83,88,87,99

第二遍(2 间隔排序):按照间隔 2 分割子数组,共分成两组,分别是

50,60,61,83,87\],\[70,80,84,88,99\]。对他们进行插入排序,排序后它们分别变成:\[50,60,61,83,87\],\[70,80,84,88,99\],此时整个数组变成 \[50,70,60,80,61,84,83,88,87,99\]。这里有一个非常重要的性质:当我们完成 2 间隔排序后,这个数组仍然是保持 5 间隔有序的。也就是说,**更小间隔的排序没有把上一步的结果变坏**。 第三遍(1 间隔排序,等于直接插入排序):按照间隔 1 分割子数组,分成一组,也就是整个数组。对其进行插入排序,经过前两遍排序,数组已经基本有序了,所以这一步只需经过少量交换即可完成排序。排序后数组变成 \[50,60,61,70,80,83,84,87,88,99\],整个排序完成。

相关推荐
田梓燊7 小时前
红黑树分析 1
算法
晚风吹长发8 小时前
二分查找算法+题目详解
c++·算法·二分查找
悠悠~飘8 小时前
18.PHP基础-递归递推算法
算法·php
pilgrim538 小时前
结合 Leetcode 题探究KMP算法
算法·leetcode
罗义凯8 小时前
其中包含了三种排序算法的注释版本(冒泡排序、选择排序、插入排序),但当前只实现了数组的输入和输出功能。
数据结构·c++·算法
kevien_G19 小时前
JAVA之二叉树
数据结构·算法
syt_biancheng9 小时前
Day3算法训练(简写单词,dd爱框框,3-除2!)
开发语言·c++·算法·贪心算法
二进制的Liao10 小时前
【编程】脚本编写入门:从零到一的自动化之旅
数据库·python·算法·自动化·bash
自然数e10 小时前
C++多线程【线程管控】之线程转移以及线程数量和ID
开发语言·c++·算法·多线程
云在Steven11 小时前
在线确定性算法与自适应启发式在虚拟机动态整合中的竞争分析与性能优化
人工智能·算法·性能优化