排序-希尔排序

介绍

希尔排序属于那种没有了解过的直接看代码一脸懵逼的,

所以同学们尽量不要直接看代码,仔细阅读本篇博客内容。

插入排序本来算是一个低效排序,

一次只可以挪动一个数据,

但是,它的强来了!!!

---Donald Shell(希尔)

对插入排序进行了优化
将插入排序提升了不止一个档次
甚至可以和快速排序平起平坐!

基本思想

先选定一个整数,把待排序文件中所有记录分成个
组,所有距离为的记录分在同一组内,并对每一组内的记录进行排序。

然后,取,重复上述分组和排序的工
作。当到达=1时,所有记录在统一组内排好序。

希尔排序法又叫做缩小增量法。

其本质就是对固定间隔组成的序列进行插入排序,

然后此固定间距步步缩小,最后进行直接插入排序

先进行预排序,再进行直接插入排序

根据上图,明显看出,进行预排序过后的数列,已经有了基本的顺序,

大大缩小了大数字在前,小数字在后还需要一个一个挪动的尴尬局面

代码实现

void ShellSort(int* a, int n)
{
    // 1、gap > 1 预排序
    // 2、gap == 1 直接插入排序

    int gap = n;
    while (gap > 1)
    {
        gap = gap / 3 + 1;  // +1可以保证最后一次一定是1
        // gap = gap / 2;
        for (int i = 0; i < n - gap; ++i)
        {
            int end = i;
            int tmp = a[end + gap];
            while (end >= 0)
            {
                if (a[end] > tmp)
                {
                    a[end + gap] = a[end];
                    end -= gap;
                }
                else
                {
                    break;
                }
            }
            a[end + gap] = tmp;
        }
    }
}

这是多组同时进行间隔gap排序

gap的取值

gap的取值方法有很多种.
但是每一种gap的取值都满足:

先大后小原则
也就是我们的预排序不止排序一次
gap会由大变小,常见的取值有:

gap=n/2 (n为数组长度)
gap=n/3 (n为数组长度)

同时取值时一定要确保最后一个值为1收尾

不过对于gap到底取什么值最合适,到现在也没有定论,因为快排的横空出世,再怎么研究希尔也比不过快排,也就没了意义

特性总结

1. 希尔排序是对直接插入排序的优化。
2. 当gap > 1时都是预排序,目的是让数组更接近于有序。

当gap == 1时,数组已经接近有序的了,这样就会很快。这样整体而言,

可以达到优化的效果。我们实现后可以进行性能测试的对比。
3. 希尔排序的时间复杂度不好计算,因为gap的取值方法很多,

导致很难去计算,因此在好些树中给出的
希尔排序的时间复杂度都不固定:

《数据结构(C语言版)》--- 严蔚敏

《数据结构-用面相对象方法与C++描述》--- 殷人昆

4. 稳定性:不稳定

好的,我可以为你解答关于希尔排序的问题。希尔排序是插入排序的一种改进算法,也被称为“缩小增量排序”。它的基本思想是将待排序的序列按照一定的间隔分成若干个子序列,对每个子序列进行插入排序,然后逐步缩小间隔,直到间隔为1,最后对整个序列进行一次插入排序希尔排序的时间复杂度与间隔序列的选择有关,最坏时间复杂度为$O(n^2)$,平均时间复杂度为$O(n \log n)$。相比于插入排序希尔排序的时间复杂度有了明显的提升,且其在实际应用中也表现得比插入排序好。 下面是希尔排序的示例代码: ```python def shell_sort(arr): n = len(arr) gap = n // 2 while gap > 0: for i in range(gap, n): temp = arr[i] j = i while j >= gap and arr[j - gap] > temp: arr[j] = arr[j - gap] j -= gap arr[j] = temp gap //= 2 return arr ``` 在这段代码中,我们首先定义了一个希尔排序的函数shell_sort,它接收一个列表arr作为参数,并返回排好序的列表。在函数中,我们首先获取列表的长度n和初始的间隔gap,然后开始循环,直到间隔为1为止。在每次循环中,我们将列表按照间隔分成若干个子序列,对每个子序列进行插入排序,将子序列按照升序排列。最后,我们将间隔除以2,继续循环直到间隔为1。 使用希尔排序对列表进行排序非常简单,只需要调用shell_sort函数即可。例如,我们可以这样调用函数: ```python arr = [5, 2, 8, 3, 1, 6] sorted_arr = shell_sort(arr) print(sorted_arr) ``` 这段代码会输出排好序的列表[1, 2, 3, 5, 6, 8]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值