数据结构与算法之希尔排序

前言

希尔排序(Shell Sort)是插入排序的一种高效改进版本,由 Donald Shell 于 1959 年提出。它的核心思想是通过分组插入排序来减少数据交换的次数,从而提升排序效率。

一、 核心思想:缩小增量排序

希尔排序引入了"增量(Gap)"的概念。它不像直接插入排序那样逐个比较相邻元素,而是将整个待排序的序列分割成若干个子序列,对每个子序列进行插入排序。随着增量逐渐减小,整个序列会变得越来越"基本有序",当增量减至 1 时,整个序列实际上就是一次直接插入排序,但此时数据已经基本有序,所以移动次数会大大减少。

二、 算法步骤

假设我们有数组:[8, 3, 1, 9, 5, 7, 2, 4, 6]为例,看一下算法排序的过程

1.选择增量序列

首先我们选择一个增量序列 gap,通常初始值取数组长度的一半(n/2)。

这里的n/2向下取整。

将元素按间隔4分组:

组1: 8, 5, 6 → 排序后: 5, 6, 8

组2: 3, 7 → 排序后: 3, 7

组3: 1, 2 → 排序后: 1, 2

组4: 9, 4 → 排序后: 4, 9

数组变为:5, 3, 1, 4, 6, 7, 2, 9, 8

2.分组排序

将数组按增量 gap分成若干组,对每组进行插入排序。

这里的间隔为2.

组1: 5, 1, 6, 2, 8 → 排序后: 1, 2, 5, 6, 8

组2: 3, 4, 7, 9 → 排序后: 3, 4, 7, 9

数组变为:1, 3, 2, 4, 5, 7, 6, 9, 8

3.缩小增量

将增量 gap缩小(通常为 gap/2gap/3),重复步骤 2。

这里的gap = 1。直接执行下一个步骤。

4.最终排序

当增量 gap缩小为 1 时,对整个数组进行一次直接插入排序,排序完成。

最终排序:1, 2, 3, 4, 5, 6, 7, 8, 9

三、 代码实现(C语言)

复制代码
#include <stdio.h>

void shellSort(int arr[], int n) {
    // 初始增量 gap 为数组长度的一半
    for (int gap = n / 2; gap > 0; gap /= 2) {
        // 从第 gap 个元素开始,逐个对其所在组进行直接插入排序
        for (int i = gap; i < n; i++) {
            int temp = arr[i]; // 待插入的元素
            int j;
            // 对组内元素进行插入排序
            for (j = i; j >= gap && arr[j - gap] > temp; j -= gap) {
                arr[j] = arr[j - gap]; // 移动元素
            }
            arr[j] = temp; // 插入元素
        }
    }
}

int main() {
    int arr[] = {9, 6, 7, 3, 1, 5, 4, 8, 2};
    int n = sizeof(arr) / sizeof(arr[0]);
    
    shellSort(arr, n);
    
    printf("排序结果: ");
    for (int i = 0; i < n; i++) {
        printf("%d ", arr[i]);
    }
    return 0;
}

四、 性能分析

特性 描述
时间复杂度 取决于增量序列的选择。最坏情况为 O(n²),使用 Hibbard 增量序列可优化至 O(n^(3/2))。
空间复杂度 O(1),是原地排序算法。
稳定性 不稳定。由于分组排序,相等的元素可能会被分到不同的组,导致相对顺序改变。

五、 总结

希尔排序是插入排序的优化版 ,它通过"先宏观调整,后微观调整 "的策略,有效减少了数据交换的次数。虽然它不如快速排序或归并排序那样高效,但它在中等规模数据的排序中表现良好,且代码实现相对简单,是理解排序算法优化思路的重要案例。

相关推荐
星马梦缘5 分钟前
算法设计与分析 作业三 纯答案
算法
吴阿福|一人公司7 分钟前
深度解析 Python 类变量修改的命名空间隔离
java·服务器·数据结构
不知名的老吴32 分钟前
经典算法题之行星碰撞
数据结构·算法
丘山望岳38 分钟前
剑起霜华——平衡二叉树(AVL树 )精讲
开发语言·数据结构·c++
西安邮电大学38 分钟前
有关数组的经典算法题
java·后端·其他·算法·面试
学Linux的语莫43 分钟前
大模型微调数据集格式详解:Alpaca、ShareGPT、DPO、KTO、预训练数据怎么构建?
人工智能·算法·机器学习·微调格式
wayz111 小时前
Momentum:UO(终极震荡指标)技术指标详解
算法·金融·数据分析·量化交易·特征工程
Boom_Shu1 小时前
浅拷贝与深拷贝
开发语言·c++·算法
LuminousCPP1 小时前
数据结构 - 单链表第一篇:单链表基础操作
c语言·数据结构·经验分享·笔记·学习
触底反弹1 小时前
一文彻底搞懂 JavaScript 栈和队列(建议收藏)
javascript·算法·面试