【数据结构排序算法篇】----希尔排序【实战演练】

简介: 【数据结构排序算法篇】----希尔排序【实战演练】

作为一名对技术充满热情的学习者,我一直以来都深刻地体会到知识的广度和深度。在这个不断演变的数字时代,我远非专家,而是一位不断追求进步的旅行者。通过这篇博客,我想分享我在某个领域的学习经验,与大家共同探讨、共同成长。请大家以开放的心态阅读,相信你们也会在这段知识之旅中找到启示。



前言

大家好!新年快乐哦,年后我们学习还要继续,博主依然要定时发文,今天我们一起来讲讲另一个排序算法----希尔排序


一、什么是希尔排序

希尔排序是一种基于插入排序的排序算法,由Donald Shell于1959年提出。其关键的概念是将原来要排序的列表划分成若干个较小的子列表,分别进行插入排序,然后逐渐减少子列表的数量,直到全列表作为一个列表来对待进行插入排序为止。这种方法中,每个子列表实际上是间隔一定"增量"的一组元素构成的。

希尔排序算法的步骤概括为:

  1. 选择一个增量序列t1, t2, ..., tk,其中ti > tjtk = 1(通常使用序列中每个元素作为n/2的结果,n是要排序的元素的数量)。
  2. 按增量序列个数k,对序列进行k轮排序。
  3. 每轮排序,根据对应的增量ti,将待排序列分割成若干长度为m的子序列,分别对各子表进行直接插入排序。仅增量因子为1时,整个序列作为一个表来处理,表长度即为整个序列的长度。

希尔排序算法的效率与增量序列的选择有很大关系。在最好的情况下,使用合适的增量序列,希尔排序的时间复杂度可以达到O(nlogn),但最坏情况下可能为O(n^2)

以下是希尔排序的一个基本实现示例:

public class ShellSort {
    public static void shellSort(int[] array) {
        int n = array.length;
        // Gaps序列以n/2开始,并逐步递减至1
        for (int gap = n / 2; gap > 0; gap /= 2) {
            // 从gap开始对所有元素进行插入排序
            for (int i = gap; i < n; i++) {
                // 添加array[i]到已排序的序列中
                int temp = array[i];
                int j;
                for (j = i; j >= gap && array[j - gap] > temp; j -= gap) {
                    array[j] = array[j - gap];
                }
                array[j] = temp;
            }
        }
    }
}

关于希尔排序的时间复杂度

希尔排序的时间复杂度是依赖于其所使用的增量序列的。目前还没有确切的公式能够描述所有情况下希尔排序的时间复杂度,因为不同的增量序列会导致算法行为大不相同。

对于最原始的希尔排序,即增量序列取值为n/2, n/4, ..., 1,平均和最坏的时间复杂度大约在O(n^1.5)左右。但这并不是最优的增量序列,随着研究的不断深入,人们发现了更好的增量序列,使得希尔排序的性能得到大幅提升。

例如,使用Hibbard的增量序列(1, 3, 7, ..., 2^k - 1)时,希尔排序的最坏情况时间复杂度是O(n^1.5)。而使用Sedgewick提出的一种增量序列的希尔排序,时间复杂度可以达到O(n^1.3)。有更复杂的增量序列,如Pratt序列,可以让希尔排序达到几乎接近O(nlogn)的性能。

除了时间复杂度,希尔排序的另一个重要性能标准是其比较次数和移动(或交换)次数。由于希尔排序在前面的步骤可能会进行大范围的移动,后面的排序步骤需要的移动和比较就会较少。

综上所述,希尔排序的时间复杂度无一定的定论,它受所选择的增量序列的影响很大,且具体情况下的性能也受到数据初始状态的影响。尽管如此,希尔排序通常比传统的插入排序快,特别是在数据量较大的情况下。

二、希尔排序相对于传统的插入排序有什么优势

希尔排序相对于传统的插入排序有几个明显的优势:

  1. 处理大量无序数据的能力: 在插入排序中,如果一个元素需要被移动到很远的位置,它必须逐步通过多次移动到达最终位置。而希尔排序允许交换相距较远的元素,使得可以一步到位地将一个元素移动到很远的位置,这对于大量无序的数组来说尤其有效。
  2. 减少逆序的对数: 希尔排序通过先进行宽间隔的排序来减少整个数组中逆序对(即不正确顺序的元素对)的数量,从而使后续的插入排序步骤变得更加高效。
  3. 更好的子数组性能: 由于希尔排序是在多个子数组上分别进行插入排序,当数组的大小缩小后,插入排序在小数组上表现是非常高效的,因为它利用了局部性原理。
  4. 适用性广: 希尔排序不需要大量的辅助空间,只需要一个额外的元素空间进行交换,因此它是一种原地排序算法。这意味着它在空间限制的环境中是非常实用的。
  5. 适度的递增序列效率非常高: 通过适当地选择递增序列,可以使希尔排序在实际应用中表现出比其他时间复杂度为O(n^2)的算法(如简单插入排序、冒泡排序等)更优秀的性能。

尽管希尔排序在理论上的最坏情况运行时间比某些高级排序算法(如归并排序、快速排序)要慢,但由于其相对简单和低开销的特点,在处理中等大小的数组时它通常会表现得非常高效。因此,在需要考虑算法实现的复杂性和排序效率时,它是一个不错的选择。

三、练习

简单习题

数组 [8, 6, 7, 9, 5]

假设我们选择的增量序列是简单的除以2策略,第一个增量为 n/2(向下取整),接着是 n/4,依此类推,直到增量减到1。

初始数组:[8, 6, 7, 9, 5]

第一轮排序(增量为 5/2 = 2,向下取整):

分两组进行排序:

  • 第一组:[8, 7]
  • 第二组:[6, 9]
    在这种情况下,因为我们的数组较小,每组数据最多只有两个元素,因此在这个步骤中,这两个子数组都被认为是有序的。

第二轮排序(增量为 2/2 = 1,即最终的插入排序):

从第二个元素开始(因为第一个元素本身已经是“有序的”,作为一元数组),将每个元素与前面的元素比较进行插入排序:

  • 比较6和8:[6, 8, 7, 9, 5] (6和8交换位置)
  • 比较7和8:[6, 7, 8, 9, 5] (7插入到8前面,因为已经比8小了)
  • 比较9和8:不需要操作,因为9大于前面的8
  • 比较5与前面的所有元素:
  • 比较5和9:[6, 7, 8, 5, 9] (5和9交换位置)
  • 比较5和8:[6, 7, 5, 8, 9] (5和8交换位置)
  • 比较5和7:[6, 5, 7, 8, 9] (5和7交换位置)
  • 比较5和6:[5, 6, 7, 8, 9] (5插入到第一个位置)

最终排序后的数组是 [5, 6, 7, 8, 9]。

这个过程演示了希尔排序如何通过首先使用较大的增量减少逆序对,然后再进行最终的插入排序,从而高效地对数组进行排序。这个例子中的数组较小,所以看起来希尔排序和插入排序差异不大,但在处理更大的数组时,希尔排序的效率提升将会更加明显。

较难习题

初始数组:[33, 10, 14, 27, 35, 19, 42, 44, 31, 3]

我们仍然可以使用递减的增量序列,这里我们选择增量序列为 n/2, n/4, …, 1(对每个增量向下取整)。

第一轮排序(增量为 10/2 = 5):

分五组进行排序:

  • 第一组:[33, 19]
  • 第二组:[10, 42]
  • 第三组:[14, 44]
  • 第四组:[27, 31]
  • 第五组:[35, 3]

对每组使用插入排序(由于只有两个元素,所以如果第二个元素比第一个小就交换它们):

  • 组1:19 比 33 小,交换它们。新数组:[19, 10, 14, 27, 3, 33, 42, 44, 31, 35]
  • 组2、组3、组4 由于都是由数目只有两个的组,而且都是有序的,所以不需要交换。
  • 组5:3 比 35 小,交换它们。新数组 (更新组5后):[19, 10, 14, 27, 3, 33, 42, 44, 31, 35]

第二轮排序(增量为 5/2 = 2):

分为两组进行排序:

  • 奇数索引组:[19, 14, 3, 42, 31]
  • 偶数索引组:[10, 27, 33, 44, 35]

对每组使用插入排序:

  • 奇数索引组变成了 [3, 14, 19, 31, 42]
  • 偶数索引组变成了 [10, 27, 33, 35, 44]

合在一起的新数组:[3, 10, 14, 27, 19, 31, 33, 35, 42, 44]

第三轮排序(逐元素插入排序,增量为 1):

使用插入排序对整个数组排序:

  • 10, 14, 27, 19, 31, 33, 35, 42, 44 都已经有序。
  • 3是最小的元素,已经在正确的位置。

最终结果没有变化,因为在上一轮中,经过奇数和偶数分组的插入排序之后,数组已经是完全有序的了。这恰好是一个较为理想的场景,在实际应用中可能很少见。实际情况下,最后一轮通常需要一些移动和比较才能将所有元素排序到最终位置。

排序后的数组:[3, 10, 14, 27, 19, 31, 33, 35, 42, 44]

注意,上面的步骤可能需要更多的交换和比较才能达到完全有序,上述过程省略了部分细节。希尔排序的效果很大程度上依赖于选择的增量序列。在不同环境和应用领域,可能会有不同的增量序列以期望获得更优的性能。

四、Java面试题

  • Java面试题:

请在Java中实现希尔排序,并解释其时间复杂度与工作原理。此外,请分析希尔排序比起直接插入排序的优势。

解答例子:

public class ShellSort {
    public static void shellSort(int[] array) {
        int n = array.length;
        
        // Start with a big gap, then reduce the gap
        for (int gap = n / 2; gap > 0; gap /= 2) {
            // Do a gapped insertion sort for this gap size.
            // The first gap elements array[0..gap-1] are already in gapped order
            // keep adding one more element until the entire array is gap sorted
            for (int i = gap; i < n; i += 1) {
                // add array[i] to the elements that have been gap sorted
                // save array[i] in temp and make a hole at position i
                int temp = array[i];
                
                // shift earlier gap-sorted elements up until the correct location for array[i] is found
                int j;
                for (j = i; j >= gap && array[j - gap] > temp; j -= gap) {
                    array[j] = array[j - gap];
                }
                
                // put temp (the original array[i]) in its correct location
                array[j] = temp;
            }
        }
    }
    public static void main(String[] args) {
        int[] exampleArray = {33, 10, 14, 27, 35, 19, 42, 44, 31, 3};
        shellSort(exampleArray);
        for (int i : exampleArray) {
            System.out.print(i + " ");
        }
    }
}

时间复杂度:

希尔排序的最坏情况和平均情况运行时间都依赖于所使用的增量序列。对于上面代码中使用的增量序列(n/2, n/4, …, 1),最坏情况时间复杂度为O(n2),但是实际的性能可能比O(n2)好一些,因为它启动时每个子列表都比较短。对于其他增量序列,特别是最好的已知序列,时间复杂度可以降至O(n log^2 n)。不过,希尔排序的确切时间复杂度仍然是一个开放的问题。

工作原理:

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

  1. 插入排序在对几乎已经排好序的数据操作时,效率高(即数据越接近最终排列的顺序越好),即可达到线性排序的效率。
  2. 但插入排序一般来说是低效的,因为插入排序每次只能将数据移动到相邻的位置,每次只能移动一步。

希尔排序的基本思想是将数组列在一个表中并对列分别进行插入排序,从而将不相邻的元素进行比较,以移动那些距离正确位置很远的元素。然后逐渐减小间隔(增量gap),继续排序,最终增量为1并执行普通的插入排序,此时文件已经基本有序,所以最后这步会很快。

优势分析:

相较于直接插入排序,希尔排序的优势在于它允许交换的项之间有较大的间隔,使得较小的项可以一次性通过多次交换达到较远的位置。因此,希尔排序对于大规模的乱序数组来说,比直接插入排序有更高的执行效率。

该面试题评估了求职者对排序算法的理解和代码实现能力,并且通过要求解释希尔排序的工作原理及优势,检验了求职者的分析和表达能力。此外,代码质量和可读性也是评估的一个重要方面。


总结

以上就是对希尔排序的初步总结,同学们还需要加强训练,对于算法的掌握就是理解原理后在不同场景下频繁的使用,才能够真正的熟练掌握数据结构这门课程。

感谢大家抽出宝贵的时间来阅读博主的博客,新人博主,感谢大家关注点赞,祝大家未来的学习工作生活一帆风顺,加油!!!

目录
相关文章
|
13天前
|
算法 数据处理 C语言
C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合
本文深入解析了C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合,旨在帮助读者掌握这一高效的数据处理方法。
23 1
|
16天前
|
机器学习/深度学习 算法 数据挖掘
K-means聚类算法是机器学习中常用的一种聚类方法,通过将数据集划分为K个簇来简化数据结构
K-means聚类算法是机器学习中常用的一种聚类方法,通过将数据集划分为K个簇来简化数据结构。本文介绍了K-means算法的基本原理,包括初始化、数据点分配与簇中心更新等步骤,以及如何在Python中实现该算法,最后讨论了其优缺点及应用场景。
58 4
|
2月前
|
存储 Java 开发者
Java Map实战:用HashMap和TreeMap轻松解决复杂数据结构问题!
【10月更文挑战第17天】本文深入探讨了Java中HashMap和TreeMap两种Map类型的特性和应用场景。HashMap基于哈希表实现,支持高效的数据操作且允许键值为null;TreeMap基于红黑树实现,支持自然排序或自定义排序,确保元素有序。文章通过具体示例展示了两者的实战应用,帮助开发者根据实际需求选择合适的数据结构,提高开发效率。
70 2
|
14天前
|
存储 算法 搜索推荐
Python 中数据结构和算法的关系
数据结构是算法的载体,算法是对数据结构的操作和运用。它们共同构成了计算机程序的核心,对于提高程序的质量和性能具有至关重要的作用
|
13天前
|
数据采集 存储 算法
Python 中的数据结构和算法优化策略
Python中的数据结构和算法如何进行优化?
|
21天前
|
算法
数据结构之路由表查找算法(深度优先搜索和宽度优先搜索)
在网络通信中,路由表用于指导数据包的传输路径。本文介绍了两种常用的路由表查找算法——深度优先算法(DFS)和宽度优先算法(BFS)。DFS使用栈实现,适合路径问题;BFS使用队列,保证找到最短路径。两者均能有效查找路由信息,但适用场景不同,需根据具体需求选择。文中还提供了这两种算法的核心代码及测试结果,验证了算法的有效性。
82 23
|
21天前
|
算法
数据结构之蜜蜂算法
蜜蜂算法是一种受蜜蜂觅食行为启发的优化算法,通过模拟蜜蜂的群体智能来解决优化问题。本文介绍了蜜蜂算法的基本原理、数据结构设计、核心代码实现及算法优缺点。算法通过迭代更新蜜蜂位置,逐步优化适应度,最终找到问题的最优解。代码实现了单链表结构,用于管理蜜蜂节点,并通过适应度计算、节点移动等操作实现算法的核心功能。蜜蜂算法具有全局寻优能力强、参数设置简单等优点,但也存在对初始化参数敏感、计算复杂度高等缺点。
56 20
|
13天前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
37 1
|
21天前
|
机器学习/深度学习 算法 C++
数据结构之鲸鱼算法
鲸鱼算法(Whale Optimization Algorithm,WOA)是由伊朗研究员Seyedali Mirjalili于2016年提出的一种基于群体智能的全局优化算法,灵感源自鲸鱼捕食时的群体协作行为。该算法通过模拟鲸鱼的围捕猎物和喷出气泡网的行为,结合全局搜索和局部搜索策略,有效解决了复杂问题的优化需求。其应用广泛,涵盖函数优化、机器学习、图像处理等领域。鲸鱼算法以其简单直观的特点,成为初学者友好型的优化工具,但同时也存在参数敏感、可能陷入局部最优等问题。提供的C++代码示例展示了算法的基本实现和运行过程。
43 0
|
21天前
|
算法 vr&ar 计算机视觉
数据结构之洪水填充算法(DFS)
洪水填充算法是一种基于深度优先搜索(DFS)的图像处理技术,主要用于区域填充和图像分割。通过递归或栈的方式探索图像中的连通区域并进行颜色替换。本文介绍了算法的基本原理、数据结构设计(如链表和栈)、核心代码实现及应用实例,展示了算法在图像编辑等领域的高效性和灵活性。同时,文中也讨论了算法的优缺点,如实现简单但可能存在堆栈溢出的风险等。
35 0