数据结构的希尔排序(c语言版)

news2024/11/27 11:53:29

一.希尔排序的概念

1.希尔排序的基本思想

希尔排序是一种基于插入排序算法的优化排序方法。它的基本思想如下:

  1. 选择一个增量序列 t1,t2,......,tk,其中 ti > tj, 当 i < j,并且 tk = 1。

  2. 按增量序列个数k,对数组进行k趟排序:

    • 第一趟,按 t1 的增量对数组进行插入排序;
    • 第二趟,按 t2 的增量对数组进行插入排序;
    • ... ...
    • 第k趟,按 tk 的增量(此时 tk = 1),也就是对整个数组进行插入排序。

2.希尔排序的优点

  1. 时间复杂度较低。希尔排序的时间复杂度一般在 O(n^1.25) 和 O(n^1.5) 之间,优于简单的插入排序。

  2. 在部分有序的数组中效率很高。希尔排序通过分组插入排序来利用数据的局部有序性,可以有效地加快排序速度。

  3. 空间复杂度低,只需要常量级的额外空间。

  4. 代码实现相对简单,易于理解和编码。

3.希尔排序的缺点

  1. 增量序列的选择对排序效率有很大影响。不同的增量序列会导致很大的性能差异。找到最优的增量序列是一个难题。

  2. 在数据量很大时,性能可能不如其他算法,如快速排序、堆排序等。

  3. 不稳定。希尔排序是不稳定的排序算法,即相等的元素可能会改变相对次序。

  4. 理论分析复杂。希尔排序的时间复杂度分析比较困难,没有得到一个统一的结论。

4.希尔排序与快速排序在使用情况时的差异 

  1. 数据量中等且部分有序:

    • 希尔排序通过分组排序利用了数据的局部有序性,在部分有序的数组上表现很好。
    • 而快速排序在部分有序数组上可能会退化为 O(n^2) 的时间复杂度。
  2. 内存受限的环境:

    • 希尔排序只需要常量级的额外空间,而快速排序需要递归调用栈,在内存受限的环境下可能会有优势。
  3. 数据分布不均匀:

    • 快速排序的性能很容易受到数据分布的影响,而希尔排序相对更加好。
  4. 预先知道数据大致分布情况:

    • 如果对数据的分布有一定了解,可以选择合适的增量序列来优化希尔排序的性能。
  5. 对稳定性要求不高:

    • 快速排序是不稳定的,而希尔排序也是不稳定的。如果稳定性不是关键,希尔排序可能是更好的选择。

二.希尔排序的功能

1.分组插入排序

  • 希尔排序的核心思想是通过分组插入排序来优化基本的插入排序算法。
  • 它首先选择一个增量序列,如 [n/2, n/4, n/8, ...],将原始数组划分为多个子数组。
  • 每个子数组的元素索引差为增量值。例如,当增量为 4 时,子数组为 arr[0]、arr[4]、arr[8]...
  • 对这些子数组分别进行插入排序。
  • 随着增量序列逐步减小,子数组中的元素越来越集中,最终整个数组被完全排序。

2.利用局部有序性

  • 在初始阶段,当增量较大时,子数组中的元素较为分散。
  • 随着增量的不断减小,子数组中的元素逐渐趋于有序。
  • 这种分组插入排序可以有效利用数据的局部有序性,从而减少插入排序的比较和移动操作次数。

3.时间复杂度优化

  • 基本插入排序的时间复杂度为 O(n^2)。
  • 而希尔排序通过分组插入排序和利用局部有序性,可以将平均时间复杂度优化到 O(n^1.25) 到 O(n^1.5)。

4.空间复杂度低

  • 希尔排序只需要常量级的额外空间来存储一些中间变量,如增量序列等。
  • 因此它的空间复杂度很低,仅为 O(1)。

5.代码实现简单

  • 与其他高效排序算法相比,如快速排序和归并排序,希尔排序的代码实现较为简单。
  • 它只需要一个嵌套循环来完成分组和插入排序即可。

三.希尔排序的代码实现

1.排序的实现

  1. 定义一个名为 shell_sort 的函数,它接受两个参数:

    • arr: 一个整型数组,需要被排序
    • n: 数组的长度
  2. shell_sort 函数的实现:

    • 使用一个 for 循环来遍历不同的增量值 gap。初始的 gap 为 n/2
    • 对于每个 gap 值,我们再次使用一个 for 循环来遍历数组。
    • 对于每个元素 arr[i],我们将其暂存到临时变量 temp 中。
    • 然后,我们使用另一个内层 for 循环来执行分组插入排序。在这个循环中,我们将 arr[j] 的值移动到 arr[j - gap] 的位置,直到找到 temp 应该插入的位置。
    • 最后,我们将 temp 赋值给 arr[j]
    • 重复上述过程,直到 gap 变为 0。
void shell_sort(int arr[], int n) {
    for (int gap = n / 2; gap > 0; gap /= 2) {
        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;
        }
    }
}

2.main函数

  • 定义一个示例数组 arr
  • 计算数组长度 n
  • 打印出未排序的数组。
  • 调用 shell_sort 函数对数组进行排序。
  • 打印出排序后的数组。
int main() {
    int arr[] = {64, 34, 25, 12, 22, 11, 90};
    int n = sizeof(arr) / sizeof(arr[0]);

    printf("Unsorted array: ");
    for (int i = 0; i < n; i++) {
        printf("%d ", arr[i]);
    }
    printf("\n");

    shell_sort(arr, n);

    printf("Sorted array: ");
    for (int i = 0; i < n; i++) {
        printf("%d ", arr[i]);
    }
    printf("\n");

    return 0;
}

3.程序的执行流程

  • 首先,在 main 函数中,我们定义了一个示例数组 arr
  • 然后,我们计算数组的长度 n
  • 接下来,我们打印出未排序的数组。
  • 调用 shell_sort 函数对数组进行排序。
  • 在 shell_sort 函数内部,我们使用一个 for 循环来迭代不同的增量值 gap。初始的 gap 为 n/2
  • 对于每个 gap 值,我们遍历数组,对于每个元素 arr[i],我们将其暂存到临时变量 temp 中。
  • 然后,我们使用另一个内层 for 循环来执行分组插入排序。在这个循环中,我们将 arr[j] 的值移动到 arr[j - gap] 的位置,直到找到 temp 应该插入的位置。
  • 最后,我们将 temp 赋值给 arr[j]
  • 重复上述过程,直到 gap 变为 0。
  • 排序完成后,我们在 main 函数中打印出排序后的数组。

四.希尔排序的源代码

  1. 首先,我们定义一个 shell_sort 函数,接受一个整型数组 arr 和数组长度 n 作为参数。
  2. 在函数内部,我们使用一个 for 循环来迭代不同的增量值 gap。初始的 gap 为 n/2
  3. 对于每个 gap 值,我们遍历数组,对于每个元素 arr[i],我们将其暂存到临时变量 temp 中。
  4. 然后,我们使用另一个内层 for 循环来执行分组插入排序。在这个循环中,我们将 arr[j] 的值移动到 arr[j - gap] 的位置,直到找到 temp 应该插入的位置。
  5. 最后,我们将 temp 赋值给 arr[j]
  6. 重复上述过程,直到 gap 变为 0。
#include <stdio.h>

void shell_sort(int arr[], int n) {
    for (int gap = n / 2; gap > 0; gap /= 2) {
        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[] = {64, 34, 25, 12, 22, 11, 90};
    int n = sizeof(arr) / sizeof(arr[0]);

    printf("Unsorted array: ");
    for (int i = 0; i < n; i++) {
        printf("%d ", arr[i]);
    }
    printf("\n");

    shell_sort(arr, n);

    printf("Sorted array: ");
    for (int i = 0; i < n; i++) {
        printf("%d ", arr[i]);
    }
    printf("\n");

    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

快速复制成功模式:解读SaaS裂变工具的核心价值

在数字化快速发展的今天&#xff0c;企业如何在竞争激烈的市场中迅速站稳脚跟&#xff0c;成为许多企业家和管理者关注的焦点。SaaS裂变工具作为一种创新的解决方案&#xff0c;以其独特的优势&#xff0c;帮助企业快速复制成功模式&#xff0c;实现业务的快速增长。 SaaS裂变工…

GTD时间管理法

Part 1. What is GTD? | 什么是GTD&#xff1f; GTD is a framework that enhances focus and productivity. Through techniques such as capturing all tasks in a trusted system and breaking down complex projects into actionable items, GTD allows individuals to co…

组件的传参等

一:组件的生命周期函数 组件的生命周期函数: created只是创建了组件内的实例对象 attached,给组件实例绑定了属性,绑定到页面节点树之后 ready准备好渲染之后,还未渲染之前 moved组件实例被移动到另一个位置后执行 detached在整个组件被被移除执行 error执行的时候,组件内…

左偏树,可合并堆

合并两个堆并维护最小或最大性质解决树上节点问题&#xff0c;从叶节点往根维护&#xff0c;每个节点看作一个堆表示到最近的叶节点的距离&#xff0c;所以每次对合并&#xff08;树高矮&#xff09;表示堆的顶点对应下标关键代码 static void dfs(int x){for(int ihead[x];i&g…

科技守护,河流水文监测保障水资源安全!

中小河流是城乡水资源的补给&#xff0c;又是不可或缺的排放渠道&#xff0c;维系着城乡水资源的平衡与生态的健康。然而&#xff0c;随着工业化、城市化的快速推进&#xff0c;河流生态环境面临着越来越大的压力。为了有效保护和合理利用河流资源&#xff0c;河流水文监测成为…

2024吉林省电赛(达盛杯)

1. 电赛F4系统板3D图 提起自制STM32F407VET6系统板 2. 电赛原理图 3. 电赛PCB图 4. 智能车实物图 下图是电赛的实物图&#xff0c;结构采用3D打印 5. 软件设计 下图是程序设计图 6. 仿真视频 (1) 变化高度 2024吉林省电赛仿真1 (2) 变化轮距 2024电赛仿真2 7. APP控制小车 …

vue项目中使用json编辑器

实现效果&#xff1a; 借助插件json-editor-vue3实现效果如图一&#xff0c;如果嫌丑可以通过类名改一下样式如图二。 实现过程&#xff1a; 安装插件&#xff1a;npm install json-editor-vue3 文档链接&#xff1a;GitCode - 开发者的代码家园 <script setup name&quo…

Django 入门教程

1. Django简介 基本介绍 Django 是一个由 Python 编写的一个开放源代码的 Web 应用框架。 MVC 与 MVT 模型 MVC 模型 MVC 模式&#xff08;Model–view–controller&#xff09;是软件工程中的一种软件架构模式&#xff0c;把软件系统分为三个基本部分&#xff1a;模型&am…

NSSCTF中的pop、babyupload、cve版本签到、奇妙的MD5、easy_html

目录 [SWPUCTF 2021 新生赛]pop [NISACTF 2022]babyupload ​编辑[GKCTF 2020]cve版签到 [SWP5UCTF 2022 新生赛]奇妙的MD5 [HNCTF 2022 Week1]easy_html 今日总结&#xff1a; [SWPUCTF 2021 新生赛]pop 1.代码审计 <?phperror_reporting(0); show_source("…

文本匹配.grep与Select-String用法对比

Linux Shell与PowerShell上匹配字符串 grep与Select-String用法对比 - 文章信息 - Author: 李俊才 (jcLee95) Visit me at CSDN: https://jclee95.blog.csdn.netMy WebSite&#xff1a;http://thispage.tech/Email: 291148484163.com. Shenzhen ChinaAddress of this article…

Lin网络二

目录 根据域名查找出IP地址的流程&#xff1a; DNS域名的功能&#xff1a; 正向解析&#xff1a; 反向解析&#xff1a; DNS的端口&#xff1a; DNS域名解析工作原理&#xff1a; DNS域名解析查询方式&#xff1a; 递归查询&#xff1a;&#xff08;简单来说就是将DNS解…

ICML 2024 | 北大、字节提出新型双层位置编码方案,有效改善长度外推效果

在这项工作中&#xff0c;我们利用语言序列的内在分段特性&#xff0c;设计了一种新的位置编码方法来达到更好的长度外推效果&#xff0c;称为双层位置编码&#xff08;BiPE&#xff09;。对于每个位置&#xff0c;我们的 BiPE 融合了段内编码和段间编码。段内编码通过绝对位置…

LOTO示波器软件新增导览功能

新版本的大部分型号LOTO示波器的上位机软件我们改成了导航工具条方式。原来的方式是把所有功能都显示在不同的标签页中&#xff0c;这样的优点是非常快捷方便&#xff0c;基本上用鼠标一两次点击就能直达想要的功能设置。但是缺点是不熟练的客户可能记不住各种功能的标签位置在…

安全攻防三

一、IDS: 当黑客绕过了防火墙&#xff0c;你该如何发现&#xff1f; IDS &#xff08;Intrusion Detection System&#xff0c;入侵检测系统&#xff09; NIDS 内网中检测网络流量攻击 黑客如果已经进去内网&#xff0c;防火墙就没办法保护了 NIDS部署在交换机和路由器这些路…

曲线拟合工具软件(免费)

曲线拟合是数据处理中经常用到的数值方法,本质是使用某一个模型(方程或者方程组)将一系列离散的数据拟合成平滑的曲线或者曲面,数值求解出对应的函数参数,大家可以利用MATLAB的曲线拟合工具箱也可以使用第三方的拟合软件,今天我们介绍Welsim免费的曲线拟合软件 1、MATLA…

视频技术在智慧营业厅中的应用:AI识别与智能化转型

一、方案背景 随着信息技术的快速发展&#xff0c;图像和视频分析技术已广泛应用于各行各业&#xff0c;特别是在营业厅场景中&#xff0c;该技术能够有效提升服务质量、优化客户体验&#xff0c;并提高安全保障水平。TSINGSEE青犀智慧营业厅视频管理方案旨在探讨视频监控和视…

Kafka之【存储消息】

数据已经由生产者Producer发送给Kafka集群&#xff0c;当Kafka接收到数据后&#xff0c;会将数据写入本地文件中。 存储组件 Kafka 的消息存储涉及多个关键组件&#xff0c;每个组件在消息的存储和管理过程中扮演着特定的角色。以下是 Kafka 存储消息过程中涉及的主要存储组件及…

linux--实时性优化

linux--实时性优化 1 介绍2 实时性需求3 代表性实时系统4 嵌入式系统嵌入式软件系统结构处理器时钟节拍多任务机制任务调度方式任务调度算法时间片调度算法优先级调度算法基于优先级的时间片调度算法 5 cyclictest 测试工具命令说明 6 linux 实时性改进某版本上发布实时补丁或者…

(三)MySQL 索引

欢迎访问 什么是索引&#xff1f; 提高查询效率的一种数据结构&#xff0c;索引是数据的目录 索引的分类 按「数据结构」分类&#xff1a;Btree索引、Hash索引、Full-text索引。按「物理存储」分类&#xff1a;聚簇索引、二级索引。按「字段特性」分类&#xff1a;主键索引…

JavaScript: Uncaught SyntaxError: Invalid or unexpected token

好久没有碰前端的代码了。。。 今天测试WebSocket功能&#xff0c;Client端&#xff1a;使用HtmlJavaScript&#xff0c;通过浏览器解析后&#xff0c;当做客户端&#xff1b; 客户端代码如下&#xff1a; <!DOCTYPE html> <html lang"en"><head&g…