目录

1、希尔排序介绍

1.1、定义

1.2、核心思想

2、希尔排序的流程

第 1 轮:gap = 4

第 2 轮:gap = 2

第 3 轮:gap = 1

3、希尔排序的实现

4、时间复杂度分析

5、希尔排序的优缺点

6、适用场景


前言

        希尔排序(Shell Sort)是一种基于插入排序的高效排序算法,由 Donald Shell 在 1959 年提出。

gap(间隔)是希尔排序的核心,它决定了如何将数组分成若干子序列进行插入排序。

  • 初始 gap 较大(如 n/2),逐步缩小,直到 gap = 1(此时就是普通插入排序)。

  • 每次按 gap 分组后,对每组进行独立的插入排序。

gap的作用:

  1. gap 的作用

    • 控制分组的间隔,让元素能“长距离跳跃”,减少后续操作次数。

  2. 子序列的划分

    • 每隔 gap 个元素取一个值,形成一组,组内进行插入排序。

  3. 为什么快?

    • 大步长让数据快速接近有序,小步长最终精细化调整。


1、希尔排序介绍

1.1、定义

        它通过引入分组间隔(gap)来优化插入排序的性能,使得元素能够更快地移动到正确的位置,从而减少比较和交换的次数。

1.2、核心思想

1、分组插入排序

将整个数组按照一定的间隔(gap)分成若干子序列,对每个子序列进行插入排序。

2、逐步缩小间隔

随着排序的进行,gap 不断缩小,直到 gap = 1(此时相当于普通的插入排序)。

3、最终排序

当 gap = 1 时,数组已经基本有序,插入排序的效率会非常高(接近 O(n))。

为什么比普通插入排序快?

        插入排序在数据基本有序时效率很高(接近 O(n)),但在完全逆序时效率很低(O(n²))。

        希尔排序通过先让数据“大致有序”再执行插入排序,从而显著提高性能。


2、希尔排序的流程

如下图所示:

示例:

  1. 选择一个 gap 序列(如 n/2, n/4, ..., 1)。

  2. 对每个 gap 值,将数组分成若干子序列,并对每个子序列进行插入排序。

  3. 逐步缩小 gap,重复上述过程,直到 gap = 1 完成最终排序。

示例(gap = 4, 2, 1):

原始数组:[8, 3, 6, 2, 1, 9, 5, 7, 4]

假设数组为 [8, 3, 6, 2, 1, 9, 5, 7, 4],长度为 n = 9
我们以 Shell 原始序列gap = n/2, n/4, ..., 1)为例:

第 1 轮:gap = 4

  • 将数组按间隔 4 分组,即每隔 4 个元素取一个元素,形成子序列:

    • 子序列 1:arr[0]arr[4]arr[8] → [8, 1, 4]

    • 子序列 2:arr[1]arr[5] → [3, 9](因为 arr[9] 越界,停止)

    • 子序列 3:arr[2]arr[6] → [6, 5]

    • 子序列 4:arr[3]arr[7] → [2, 7]

  • 对每个子序列进行插入排序

    • [8, 1, 4] → [1, 4, 8]

    • [3, 9] → [3, 9](已有序)

    • [6, 5] → [5, 6]

    • [2, 7] → [2, 7](已有序)

  • 排序后数组
    将子序列按原位置写回数组:

    • arr[0]=1arr[4]=4arr[8]=8 → [1, 3, 5, 2, 4, 9, 6, 7, 8]

第 2 轮:gap = 2

  • 按间隔 2 分组

    • 子序列 1:arr[0]arr[2]arr[4]arr[6]arr[8] → [1, 5, 4, 6, 8]

    • 子序列 2:arr[1]arr[3]arr[5]arr[7] → [3, 2, 9, 7]

  • 插入排序

    • [1, 5, 4, 6, 8] → [1, 4, 5, 6, 8](交换 5 和 4)

    • [3, 2, 9, 7] → [2, 3, 7, 9](交换 3 和 2,然后 9 和 7)

  • 排序后数组[1, 2, 4, 3, 5, 7, 6, 9, 8]

第 3 轮:gap = 1

  • 此时就是普通插入排序,但数组已基本有序:

    • 从 i=1 开始,逐个将元素插入到左侧已排序部分。

    • 最终结果:[1, 2, 3, 4, 5, 6, 7, 8, 9]


3、希尔排序的实现

代码示例如下:

import java.util.Arrays;public class ShellSort {/*** 希尔排序(Shell Sort)* @param arr 待排序数组*/public static void shellSort(int[] arr) {if (arr == null || arr.length <= 1) {return; // 如果数组为空或长度≤1,无需排序}int n = arr.length;// 1. 初始化间隔(gap),这里使用 Shell 原始序列:n/2, n/4, ..., 1for (int gap = n / 2; gap > 0; gap /= 2) {// 2. 对每个子序列进行插入排序(从 gap 开始,逐步向右扫描)for (int i = gap; i < n; i++) {int temp = arr[i]; // 当前待插入元素int j;// 3. 插入排序逻辑:比 temp 大的元素向后移动for (j = i; j >= gap && arr[j - gap] > temp; j -= gap) {arr[j] = arr[j - gap]; // 较大的元素后移}// 4. 将 temp 插入到正确位置arr[j] = temp;}// (可选)打印每轮排序后的数组,方便理解过程System.out.println("Gap = " + gap + ": " + Arrays.toString(arr));}}public static void main(String[] args) {int[] arr = {8, 3, 6, 2, 1, 9, 5, 7, 4};System.out.println("原始数组: " + Arrays.toString(arr));shellSort(arr);System.out.println("排序后数组: " + Arrays.toString(arr));}
}

关键步骤说明

  1. 初始化间隔(gap)

    • 初始 gap = n / 2(Shell 原始序列),之后每次缩小为 gap / 2,直到 gap = 1

    • 例如,数组长度 n = 9,则 gap 序列为 4 → 2 → 1

  2. 子序列插入排序

    • 从 i = gap 开始,逐步向右扫描,对每个子序列进行插入排序。

    • 示例gap = 4):

      • 子序列 1:arr[0]arr[4]arr[8](即 8, 1, 4 → 排序后 1, 4, 8

      • 子序列 2:arr[1]arr[5](即 3, 9 → 排序后 3, 9

      • 子序列 3:arr[2]arr[6](即 6, 5 → 排序后 5, 6

      • 子序列 4:arr[3]arr[7](即 2, 7 → 排序后 2, 7

  3. 插入排序逻辑

    • 类似普通插入排序,但步长是 gap 而不是 1

    • 如果 arr[j - gap] > temp,则向后移动元素。

  4. 插入最终位置

    • 将 temp 放到正确的位置 arr[j]

输出:

原始数组: [8, 3, 6, 2, 1, 9, 5, 7, 4]
Gap = 4: [1, 3, 5, 2, 4, 9, 6, 7, 8]
Gap = 2: [1, 2, 4, 3, 5, 7, 6, 9, 8]
Gap = 1: [1, 2, 3, 4, 5, 6, 7, 8, 9]
排序后数组: [1, 2, 3, 4, 5, 6, 7, 8, 9]


4、时间复杂度分析

如下所示:

常见 gap 序列的影响:

  • Shell 原始序列(n/2, n/4, ..., 1):最坏 O(n²)。

  • Hibbard 序列(1, 3, 7, 15, ..., 2^k -1):最坏 O(n^(3/2))。

  • Knuth 序列(1, 4, 13, 40, ..., (3^k -1)/2):平均 O(n^(3/2))。

        Java 的 Arrays.sort() 在特定情况下会使用希尔排序的变种(如 TimSort 结合插入排序优化)。


5、希尔排序的优缺点

1、优点

  • 比普通插入排序快,尤其是对中等规模数据(n ≤ 10⁴)。

  • 原地排序,空间复杂度 O(1)。

  • 适用于部分有序数据,性能接近 O(n)。

2、缺点

  • 不稳定排序(可能改变相同元素的相对顺序)。

  • 时间复杂度依赖 gap 序列,选择不当可能退化到 O(n²)。


6、适用场景

  • 中小规模数据排序(比插入排序更快,比快速排序/归并排序更节省内存)。

  • 嵌入式系统或内存受限环境(因为它是原地排序)。

  • 部分有序数据(性能接近线性时间)。


总结

  • 希尔排序是插入排序的优化版本,通过分组排序减少元素移动次数。

  • 时间复杂度介于 O(n log n) ~ O(n²),取决于 gap 序列的选择。

  • 适用于中小规模数据,在特定情况下比快速排序/归并排序更高效。

如果你需要中等规模数据进行排序,并且希望节省内存希尔排序是一个不错的选择!


参考文章:

1、六大排序算法:插入排序、希尔排序、选择排序、冒泡排序、堆排序、快速排序-CSDN博客https://blog.csdn.net/weixin_50886514/article/details/119045154?ops_request_misc=%257B%2522request%255Fid%2522%253A%25220faf03d22b2d125d5f49a4649ad59c85%2522%252C%2522scm%2522%253A%252220140713.130102334..%2522%257D&request_id=0faf03d22b2d125d5f49a4649ad59c85&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~all~top_positive~default-1-119045154-null-null.142^v102^control&utm_term=%E5%86%92%E6%B3%A1%E6%8E%92%E5%BA%8F&spm=1018.2226.3001.4187

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/90911.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/90911.shtml
英文地址,请注明出处:http://en.pswp.cn/bicheng/90911.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

c++加载qml文件

这里展示了c加载qml文件的三种方式以及qml文件中根节点的访问准备在创建工程的初期&#xff0c;遇到了一个问题&#xff0c;cmake文件以前都是系统自动生成的&#xff0c;不需要我做过多的操作修改&#xff0c;但是&#xff0c;加载qml的程序主函数是需要用到QGuiApplication&a…

007TG洞察:GPT-5前瞻与AI时代竞争力构建:技术挑战与落地路径

最近&#xff0c;GPT-5 即将发布的消息刷爆了科技圈&#xff0c;更让人期待的是&#xff0c;GPT-6 已经悄悄启动训练了&#xff0c;OpenAI 的奥特曼表示对未来1-2年的模型充满信心&#xff0c;预测AI将进化为能够发现新知识的“AI科学家”。面对日益强大的通用AI&#xff0c;企…

Windows下编译OpenVDB

本文记录在Windows下编译OpenVDB的流程。 零、环境 操作系统Windows 11VS Code1.92.1Git2.34.1MSYS2msys2-x86_64-20240507Visual StudioVisual Studio Community 2022CMake3.22.1 一、编译 1.1 下载 git clone https://github.com/AcademySoftwareFoundation/openvdb.git …

react 内置hooks 详细使用场景,使用案例

useState场景&#xff1a;组件中管理局部状态&#xff0c;如表单值、开关、计数器等。const [count, setCount] useState(0); return <button onClick{() > setCount(count 1)}>Click {count}</button>;useEffect 场景&#xff1a;组件挂载时执行副作用&#…

从0到1学Pandas(九):Pandas 高级数据结构与操作

目录一、探秘多级索引1.1 创建多级索引1.2 多级索引操作1.3 索引转换二、探索 Panel 与 xarray2.1 Panel 数据结构2.2 xarray 库2.3 高维数据操作三、时间序列高级应用3.1 时区处理3.2 时间序列重采样与频率转换3.3 时间序列分解与预测四、数据透视与重塑高级技巧4.1 复杂透视表…

C# 图像转换实战:Bitmap 转 BitmapSource 的 2 种方法

C# 图像转换实战:Bitmap 转 BitmapSource 的 2 种方法 引言 两种转换方法的完整实现 1. 基于GDI句柄的直接转换 (ToBitmapSourceFast) 2. 基于内存流的编码转换 (ToBitmapSourceSafe) 方法对比与选型指南 避坑指南 GDI句柄泄漏问题 图像显示不完整 多线程访问图像引发异常 不同…

Spring Boot 整合 Spring MVC:自动配置与扩展实践

Spring MVC 作为 Java Web 开发的核心框架&#xff0c;在传统 SSM 项目中需要大量 XML 配置&#xff08;如 DispatcherServlet、视图解析器等&#xff09;。而 Spring Boot 通过 "自动配置" 特性&#xff0c;简化了 Spring MVC 的整合过程&#xff0c;同时保留了灵活…

print(“\033[31m红\033[32m绿\033[34m蓝\033[0m默认色“)

可以让python的终端字体有着不一样的颜色。代码&#xff1a;print("\033[31m红\033[32m绿\033[34m蓝\033[0m默认色")效果&#xff1a;

LNMP-zblog分布式部署

一、准备3台主机&#xff08;rocky8&#xff09;下载相应服务[rootnginx ~]# yum install -y nginx nfs-utils[rootphp ~]# yum install -y nfs-utils php-mysqlnd php php-fpm[rootmysql ~]# yum install -y mysql-server二、挂载php端[rootphp ~]# vim /etc/exports [rootphp…

常见代码八股

1. 利用梯度下降法&#xff0c;计算二次函数yx^2x4的最小值 def target_function(x):return x ** 2 x 4def gradient(x):return 2*x 1x_init 10 x x_init steps 100 lr 0.1 for i in range(100):x x - lr*gradient(x)print(f"最小值 f(x) {target_function(x):.4f…

【深入底层】C++开发简历4+4技能描述6

简历书写 熟悉C的封装、继承、多态&#xff0c;STL常用容器&#xff0c;熟悉C11的Lambda表达式、智能指针等&#xff0c;熟悉C20协程语法&#xff0c;具有良好的编码习惯与文档能力。 回答思路 这里是基本上就是要全会&#xff0c;考察的问题也很固定&#xff0c;stl这块可以定…

forest远程调用注意事项

1、如果在项目中&#xff0c;同时依赖了其中多个框架&#xff0c;那么按 Fastjson2 > Fastjson > Jackson > Gson 这样的优先级来判断&#xff0c;Forest 会以优先级最高的框架作为 JSON 转换器。2、Forest 支持哪几种 JSON 框架&#xff1f;A: 支持 Jackson、Gson、F…

网络资源模板--基于Android Studio 实现的新闻App

目录 一、测试环境说明 二、项目简介 三、项目演示 四、部设计详情&#xff08;部分) 登录页 首页 五、项目源码 一、测试环境说明 电脑环境 Windows 11 编写语言 JAVA 开发软件 Android Studio (2020) 开发软件只要大于等于测试版本即可(近几年官网直接下载也可…

通过Location API精准获取位置信息并优化定位精度!

&#x1f44b; 你好&#xff0c;欢迎来到我的博客&#xff01;我是【菜鸟不学编程】    我是一个正在奋斗中的职场码农&#xff0c;步入职场多年&#xff0c;正在从“小码农”慢慢成长为有深度、有思考的技术人。在这条不断进阶的路上&#xff0c;我决定记录下自己的学习与成…

构建可扩展的状态系统:基于 ArkTS 的模块化状态管理设计与实现

摘要 在 HarmonyOS 的日常开发中&#xff0c;很多人都会遇到一个问题&#xff1a;多个页面之间的数据状态如何共享&#xff1f;尤其是在组件结构越来越复杂的场景下&#xff0c;如果还用传统方式来传值&#xff0c;不仅代码混乱&#xff0c;维护也很吃力。 为了解决这个问题&am…

重生之我在暑假学习微服务第二天《MybatisPlus-下篇》

本系列参考黑马程序员微服务课程&#xff0c;有兴趣的可以去查看相关视频&#xff0c;本系列内容采用渐进式方式讲解微服务核心概念与实践方法&#xff0c;每日更新确保知识点的连贯性。通过系统化学习路径帮助开发者掌握分布式系统构建的关键技术。读者可通过平台订阅功能获取…

系统整理Python的条件语句和常用方法

Python 的条件语句&#xff08;if 语句&#xff09;是控制程序流程的基础之一&#xff0c;结构清晰、语法简洁&#xff0c;非常适合初学者掌握。一、基本语法结构if 条件:执行代码块1 elif 条件2:执行代码块2 else:执行代码块3示例&#xff1a;score 85if score > 90:print…

记录个IAR程序下载后硬件复位不运行,必须断电复位才运行的问题

【问题测试】有个F407的跑马灯的例子&#xff0c;是MDK和IAR两个版本&#xff0c;MDK版本的例子下载并复位后可以正常看到LED闪烁&#xff0c;而IAR的例子下进去后&#xff0c;不会闪烁。使用TOOL的上位机内核寄存器监测工具测试发现&#xff0c;硬件复位后竟然还在调试状态&am…

观察者模式(Observer Pattern)和 发布-订阅模式(Publisher-Subscriber Pattern)

你对 观察者模式&#xff08;Observer Pattern&#xff09;和 发布-订阅模式&#xff08;Publisher-Subscriber Pattern&#xff09;的描述是非常准确的&#xff0c;并且阐明了它们的核心区别。为了帮助你更好地理解这两者的细微差异&#xff0c;下面是一个更详细的对比分析&am…

2025年接口技术的十字路口:当MCP遇见REST、GraphQL与gRPC

在当今这个由数据驱动、万物互联的时代&#xff0c;应用程序接口&#xff08;API&#xff09;已成为现代软件架构的基石。它们是不同服务之间沟通的桥梁&#xff0c;支撑着从网页应用到复杂的微服务生态系统的一切。长久以来&#xff0c;开发者们在REST、GraphQL和gRPC这几种主…