排序算法:希尔排序

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\],整个排序完成。

相关推荐
lxl13072 分钟前
C++算法(15)BFS_FloodFill
算法·宽度优先
小王C语言5 分钟前
【基础IO】————简单设计一下libc库
前端·数据结构·算法
亦复何言??6 分钟前
BeyondMimic 论文解析
人工智能·算法·机器人
WolfGang00732110 分钟前
代码随想录算法训练营 Day20 | 回溯算法 part02
算法
YXXY31312 分钟前
前缀和算法
算法
客卿12313 分钟前
滑动窗口--模板
java·算法
_日拱一卒31 分钟前
LeetCode:滑动窗口的最大值
数据结构·算法·leetcode
codeの诱惑41 分钟前
推荐算法(一):数学基础回顾——勾股定理与欧氏距离
算法·机器学习·推荐算法
落樱弥城42 分钟前
Vulkan Compute 详解
算法·ai·图形学
Book思议-43 分钟前
【数据结构】字符串模式匹配:暴力算法与 KMP 算法实现与解析
数据结构·算法·kmp算法·bf算法