图解希尔排序——希尔排序算法(shell sort)

简介: 图解希尔排序——希尔排序算法(shell sort)

希尔排序又叫缩小增量排序,它是对直接插入排序算法的一种改进。希尔排序算法的基本思想是先将整个待排序的序列划分为若干个子序列,然后分别对子序列排序,逐步缩小划分子序列的间隔,直到划分间隔为1时排序完成。

算法图解

首先给定一个序列[7, 26, 9, 11, 23, 14]

第一趟排序: 按间隔为3划分为三个子序列,对三个子序列分别进行排序,满足从小到大则不动,否则交换两个元素位置。

第二趟排序: 按间隔为2划分子序列,共划分出两个子序列,对两个子序列分别排序

第三趟排序: 按间隔为1划分进行排序

在希尔排序中有一个关键问题就是如何选取划分间隔,一种常用的间隔划分方法是:首先间隔取原始序列长度的一半,在后续排序过程中,后一趟排序的间隔为前一趟排序的间隔的一半。实际上,在业界有一个统一的公式来求划分间隔:gap = gap / 3 + 1。

算法实现(C语言)

void shell_sort(int array[], int len) 
{
  int i = 0, j = 0, k = 0, temp = 0, gap = 0;
  gap = len;
  do
  {
    gap = gap / 3 + 1;
    for(i = gap; i < len; i += gap)
    {
      k = i;
      temp = array[k];
      for(j = i - gap; (j >= 0) && (array[j] > temp); j -= gap)
      {
        array[j + gap] = array[j];
        k = j;
      }
      array[k] = temp;
    }
  }
  while(gap > 1);
}

算法分析

稳定性分析

根据上面的图示可以看出,希尔排序是不稳定的,因为希尔排序是划分子序列对子序列分别排序,如果不同子序列中有相等的数,但它们不在同一子序列中,可能会因为子序列内部排序而改变原有的顺序。比如下图所示,原始序列中有两个11,经过一趟希尔排序后,原本在后面的紫色11跑到了前面位置,所以希尔排序是不稳定的。

复杂度分析

希尔排序并不是简单的随意分组,然后各自排序,希尔排序的精髓在于按照某个增量来划分子序列(最后一个增量必须是1),实现跳跃式移动来提高排序效率,而也正是这种跳跃性带来了不稳定性。

希尔排序的时间复杂度是O(nlogn),希尔排序是最早突破排序算法复杂度O(n×n)的算法之一,从此排序算法时间复杂度不可突破O(n×n)的说法也不攻自破,并且分组划分的思想为后来很多更高效的排序算法提供了思路。

相关文章
|
1月前
|
算法 搜索推荐 Shell
数据结构与算法学习十二:希尔排序、快速排序(递归、好理解)、归并排序(递归、难理解)
这篇文章介绍了希尔排序、快速排序和归并排序三种排序算法的基本概念、实现思路、代码实现及其测试结果。
20 1
|
5月前
|
算法 搜索推荐
数据结构算法--6 希尔排序和计数排序
**希尔排序**是插入排序的改进版,通过分组插入来提高效率。它逐步减少元素间的间隔(增量序列),每次对每个间隔内的元素进行插入排序,最终增量为1时进行最后一次直接插入排序,实现整体接近有序到完全有序的过程。例如,对数组`5, 7, 4, 6, 3, 1, 2, 9, 8`,先以间隔`d=4`排序,然后`d=2`,最后`d=1`,完成排序。计数排序则适用于0到100的数值,通过统计每个数出现次数,创建对应计数数组,再根据计数重建有序数组,时间复杂度为`O(n)`。
|
3月前
|
搜索推荐 算法 Java
现有一个接口DataOperation定义了排序方法sort(int[])和查找方法search(int[],int),已知类QuickSort的quickSort(int[])方法实现了快速排序算法
该博客文章通过UML类图和Java源码示例,展示了如何使用适配器模式将QuickSort类和BinarySearch类的排序和查找功能适配到DataOperation接口中,实现算法的解耦和复用。
40 1
现有一个接口DataOperation定义了排序方法sort(int[])和查找方法search(int[],int),已知类QuickSort的quickSort(int[])方法实现了快速排序算法
|
4月前
|
算法 搜索推荐 Shell
|
5月前
|
人工智能 搜索推荐 JavaScript
心得经验总结:排序算法:插入排序法(直接插入法和希尔排序法)
心得经验总结:排序算法:插入排序法(直接插入法和希尔排序法)
36 0
|
5月前
|
机器学习/深度学习 搜索推荐 算法
【C/排序算法】:直接插入排序和希尔排序
【C/排序算法】:直接插入排序和希尔排序
45 0
|
5月前
|
搜索推荐
排序算法---希尔排序---详解&&代码
排序算法---希尔排序---详解&&代码
|
5月前
|
算法 Shell C语言
数据结构与算法——希尔排序(引例、希尔增量序列、原始希尔排序、代码、时间复杂度、Hibbard增量序列、Sedgewick增量序列)
数据结构与算法——希尔排序(引例、希尔增量序列、原始希尔排序、代码、时间复杂度、Hibbard增量序列、Sedgewick增量序列)
57 0
|
6月前
|
算法 前端开发 搜索推荐
前端算法之希尔排序
前端算法之希尔排序
35 0
|
6月前
|
搜索推荐 算法 Java
sort-06-shell sort 希尔排序算法详解
这是一个关于排序算法的系列文章摘要。文章汇总了各种排序算法,包括冒泡排序、快速排序、选择排序、堆排序、插入排序、希尔排序、归并排序、计数排序、桶排序以及大文件外部排序。特别地,希尔排序是一种改进的插入排序,通过使用不同的步长对元素进行分组排序,以提高效率。算法最终以较小的步长进行排序,接近线性时间复杂度。文章还提供了Java代码实现,并举例说明了希尔排序的过程。所有内容可在开源项目[https://github.com/houbb/sort](https://github.com/houbb/sort)中找到。