定义

希尔排序(Shellsort),也称递减增量排序算法,是插入排序的一种更高效的改进版本。希尔排序是非稳定排序算法。

希尔排序是基于插入排序的以下两点性质而提出改进方法的:

  1. 插入排序在对几乎已经排好序的数据操作时,效率高,即可以达到线性排序的效率;
  2. 但插入排序一般来说是低效的,因为插入排序每次只能将数据移动一位。

算法实现逻辑

希尔排序通过将比较的全部元素分为几个区域来提升插入排序的性能。这样可以让一个元素可以一次性地朝最终位置前进一大步。然后算法再取越来越小的步长进行排序,算法的最后一步就是普通的插入排序,但是到了这步,需排序的数据几乎是已排好的了(此时插入排序较快)。

步长序列

步长的选择是希尔排序的重要部分。只要最终步长为1任何步长序列都可以工作。算法最开始以一定的步长进行排序。然后会继续以一定步长进行排序,最终算法以步长为1进行排序。当步长为1时,算法变为普通插入排序,这就保证了数据一定会被排序。

Donald Shell最初建议步长选择为n/2并且对步长取半直到步长达到1。虽然这样取可以比O(n^2)类的算法(插入排序)更好,但这样仍然有减少平均时间和最差时间的余地。

JavaScript代码实现希尔排序

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
function shellSort(list) {
const len = list.length
let gap = len / 2 | 0
while (gap >= 1) {
for (let outer = gap; outer < len; outer++) {
const TEMP = list[outer]
let inner = outer
while (inner - gap >= 0 && list[inner - gap] > TEMP) {
list[inner] = list[inner - gap]
inner -= gap
}
list[inner] = TEMP
}
gap = gap / 2 | 0
}
return list
}