8.希尔排序

推荐视频https://www.bilibili.com/video/av17004970/?from=search&seid=17055254545953111032
非常形象的演示了希尔排序的过程.
希尔排序尝试与之前h个元素进行比较,这样通过将h从一个很大的值逐渐缩小到1,一步步将一个无序数组变成有序性更强的数组。

参考zju-mooc-数据结构


图中的注意是说,比如经过3间隔排序后的数组 依然满足5间隔有序,同理经过1间隔排序后的数组依然是满足3间隔有序的。



这个例子是说8间隔有序的话,那么4,2都可能有序,那么进行8,4,2间隔的排序就是一种浪费.

void shellSort(T arr[], int n){
    //根据给定的数组大小n来计算对应递增序列中的最大值
    int h = 1;
    while( h < n/3 )
        h = h * 3 + 1;
    /*
    *   其实在这里我觉得应该添加这样一段代码:
    *   if(n == h)
    *       h /= 3;
    *   因为当n等于h的话,进入下面的循环后 for( int i = h; i < n; i++ ) 会退出
    *   不过知道就行了,实际上对性能没什么影响
    */
    // 计算 increment sequence: 1, 4, 13, 40, 121, 364, 1093...
    while( h >= 1 ){
        for( int i = h; i < n; i++ ){
            // 对 arr[i], arr[i-h], arr[i-2*h], arr[i-3*h]... 使用插入排序
            T e = arr[i];
            int j;
            for(j = i; j >= h && e < arr[j-h]; j -= h)
                arr[j] = arr[j-h];
            arr[j] = e;
        }
        h /= 3;
    }
}

其他与前次代码相同,测试结果为:



希尔排序非常高效.

希尔排序时间复杂度的分析是比较难的,我们选取不同的让h递减的序列,它的复杂度也是不同的.
increment sequence: 1, 4, 13, 40, 121, 364, 1093...使用这个序列 时间复杂度为O(n^3/2)
希尔排序的时间复杂度比O(n^2)要低,但比O(nlogn)要高一些,不过它的实现比较简单.
不过对于排序算法来讲它的最优时间复杂度是O(nlogn)这个级别的.

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    蚁前阅读 10,589评论 0 52
  • 概述:排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    每天刷两次牙阅读 9,086评论 0 15
  • 1.插入排序—直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到已排序好...
    依依玖玥阅读 5,025评论 0 2
  • 1 初级排序算法 排序算法关注的主要是重新排列数组元素,其中每个元素都有一个主键。排序算法是将所有元素主键按某种方...
    深度沉迷学习阅读 5,349评论 0 1
  • 用来做手帐的 敲~实用边框~~~~
    沐水杰杰杰阅读 1,462评论 1 1