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

一.希尔排序的概念

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;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

HCIP的学习(25)

VLAN间通讯技术 使用多臂路由的方式 ​ 路由器的物理接口默认是不识别802.1Q标签的&#xff0c;所以&#xff0c;交换机连接路由器的接口在发送数据帧时&#xff0c;应该将标签剥离。----一般常使用Access接口配置。 单臂路由 ​ 所谓的单臂路由&#xff0c;实际上试讲路由器…

新书速览|Golang+Vue.js商城项目实战

架构师一步一步教你做项目&#xff0c;从架构设计到技术实现完整解析 本书内容 《GolangVue.js商城项目实战》以Gin和Vue.js为核心框架&#xff0c;以全栈商城项目开发为主线&#xff0c;详尽介绍前后端分离架构开发Web网站项目的关键阶段和技术细节。全书共9章&#xff0c;第…

IGMP——组播成员端网络协议

目录 一.IGMP基本概念 &#xff08;1&#xff09;组播转发困境 &#xff08;2&#xff09;感知组播成员方式 &#xff08;3&#xff09;IGMP版本 二.IGMP各版本的区别与联系 &#xff08;1&#xff09;IGMPV1 1.普遍组查询报文 2.成员关系报告报文 3.IGMPV1报文格式 4…

手撕C语言题典——返回倒数第 k 个节点(面试题)

前言 依旧力扣&#xff0c;这道题之前有做过类似的题&#xff0c;今天给一个新的思路去做&#xff0c;应对面试时候遇到的奇奇怪怪的问题 面试题 02.02. 返回倒数第 k 个节点 - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/kth-node-from-end-of-list-…

display ospf routing类型字段:Transit、Stub(区域内部)与Type2

4. B 通过 display ospf routing 命令可显示本路由器中 OSPF 路由表的信息。 其中的路由条目中 &#xff0c; 包含了 到区域内与本路由器直连 (邻居 )的路由器的路由 (TypeTransit) 、 到区域内与本路由器不直连的路由器的路由 (Type-Stub) 、 到自治系统内其它区域的路由(…

Golang | Leetcode Golang题解之第103题二叉树的锯齿形层序遍历

题目&#xff1a; 题解&#xff1a; func zigzagLevelOrder(root *TreeNode) (ans [][]int) {if root nil {return}queue : []*TreeNode{root}for level : 0; len(queue) > 0; level {vals : []int{}q : queuequeue nilfor _, node : range q {vals append(vals, node.V…

Servlet跳转404(解决)

1.解决无法跳转的404问题&#xff08;最根本&#xff0c;最重要&#xff09; 查看Project Structure&#xff0c;检查你的JDK版本不要选错版本&#xff1b; 2.页面跳转&#xff0c;url栏输入的是web.xml中的url-pattern内容&#xff0c;请仔细检查 3.关于配置信息Applicatio…

jQuery下载教程

官网&#xff1a;https://jquery.com/ ** ** 点击为压缩版本 将网站打开 界面上邮件保存为js文件即可 在html文件中引入即可 <html> <head></head> <body><script src"./js/jquery-3.6.3.js"> </script> </body> <…

卧槽!这项目开源了!【送源码 】

随着科技的飞速发展&#xff0c;个人财务管理变得越来越重要。一个名为‘Maybe’的创新型个人财务与财富管理应用程序随之诞生&#xff0c;它以其丰富的功能和用户友好的界面受到了广大用户的关注。 现在项目方将这个价值 100万美元的个人理财应用项目开源了 Maybe Maybe应用…

短剧解说一键生成原创文案的快速方法

如今短剧创作火的一塌糊涂&#xff0c;它们以其简洁明了的剧情、生动有趣的角色和紧凑的节奏&#xff0c;吸引了大量观众的关注。因此&#xff0c;它所带来的流量是非常巨大&#xff0c;不少人将流量的获取瞄准了短剧创作领域以及短剧解说领域。而对于短剧解说人员来讲&#xf…

网工内推 | 高校、外企网工,IE认证优先,年薪最高18w

01 上海外国语大学贤达经济人文学院 &#x1f537;招聘岗位&#xff1a;高校网络主管 &#x1f537;职责描述&#xff1a; 1、负责总机房、网络规划及管理&#xff0c;包括容量规划、成本评估、建设管理等; 2、负责设计、实施及维护全网络架构及规划网络变更计划 3、负责网络功…

Linux下的权限

目录 1.shell命令以及运行原理 1.1原理上初步理解shell外壳 1.1.1为什么要有shell外壳 1.1.2shell外壳是什么 1.1.3怎么办&#xff08;shell外壳的基本运行原理&#xff09; 2.Linux下的用户 3.Linux权限管理 3.1.文件访问者的分类&#xff08;人&#xff09; 3.2…

C语言 数组—— 一维数组下标越界问题分析

目录 数组元素的访问 一维数组元素的越界访问 二维数组元素的越界访问 小结 数组元素的访问 访问数组元素时&#xff0c; 下标越界 是大忌&#xff01;  编译器通常不检查下标越界&#xff0c;导致程序运行时错误  下标越界&#xff0c;将访问数组以外的空间  …

vscode插件-03 PHP

PHP Intelephense 如果php在远程计算机上&#xff0c;要把插件安装在远程&#xff0c;而不是本地。 这个插件&#xff0c;要求php版本大于7&#xff0c;且设置环境变量&#xff08;好像不一定要设置&#xff09;。 设置里面搜索php.executablePath&#xff0c;打开setting.js…

element-ui 实现输入框下拉树组件(2024-05-23)

用element-ui的 el-input&#xff0c;el-tree&#xff0c;el-popover组件组合封装 import url("//unpkg.com/element-ui2.15.14/lib/theme-chalk/index.css"); <script src"//unpkg.com/vue2/dist/vue.js"></script> <script src"//…

索引下推详情-简单入手

一.概念 索引下推&#xff08;Index Pushdown&#xff09;MySQL5.6添加的&#xff0c;是一种优化技术&#xff0c;用于在查询执行时将部分计算移动到存储引擎层&#xff0c;从而减少数据传输和计算的开销&#xff08;减少回表查询次数&#xff09;&#xff0c;提高查询性能。 …

【软件工程】【23.04】p1

关键字&#xff1a; 软件模型、提炼、加工表达工具、通信内聚、访问依赖、边界类交互分析、RUP核心工作流、首先测试数据流、软件验证过程、CMMI过程域分类工程类&#xff1b; 软件工程目的、功能需求是需求的主体、结构化方法、耦合、详细设计工具、类、类图、RUP采用用例技…

LeetCode/NowCoder-栈和队列OJ练习

孜孜不倦&#xff1a;孜孜&#xff1a;勤勉&#xff0c;不懈怠。指工作或学习勤奋不知疲倦。&#x1f493;&#x1f493;&#x1f493; 目录 说在前面 题目一&#xff1a;括号匹配问题 题目二&#xff1a;用队列实现栈 题目三&#xff1a;用栈实现队列 题目四&#xff1a;设…

不同网段的通信过程

这里的AA和HH指的是mac地址&#xff0c;上面画的是路由器 底下的这个pc1&#xff0c;或者其他的连接在这里的pc&#xff0c;他们的默认网关就是路由器的这个192.168.1.1/24这个接口 来看看通信的过程 1、先判断&#xff08;和之前一样&#xff09; 2、去查默认网关&#xf…

Matlab读取Swarm球谐系数,并绘制EWH全球格网图(存在疑问)

ICGEM官网下载 COST-G发布的4040的球谐系数 close all; clearvars -except; % addpath(E:\Code\Tool\Function\GRACE_functions); dir_degree_1 E:\Code\GRACE_data\Degree_1\deg1_coef.txt; dir_c20 E:\Code\GRACE_data\Degree_2\C20_RL06.txt; myDir_Swarm E:…