【数据结构】手撕排序(排序的概念及意义、直接插入和希尔排序的实现及分析)

目录

一、排序的概念及其运用 

1.1排序的概念 

1.2排序运用

1.3 常见的排序算法 

二、插入排序

2.1基本思想: 

2.2直接插入排序: 

2.3步骤:

2.4直接插入排序的实现

三、希尔排序( 缩小增量排序 ) 

3.1希尔排序的发展历史

3.2 希尔排序的思路

​编辑

gap = 3的思路讲解

3.3 如何选择希尔增量

四、希尔排序的代码实现


一、排序的概念及其运用 

1.1排序的概念 

排序:所谓排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起
来的操作。
稳定性:假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记
录的相对次序保持不变,即在原序列中,r[i]=r[j],且r[i]在r[j]之前,而在排序后的序列中,r[i]仍
在r[j]之前,则称这种排序算法是稳定的;否则称为不稳定的。
内部排序:数据元素全部放在内存中的排序。
外部排序:数据元素太多不能同时放在内存中,根据排序过程的要求不能在内外存之间移动数据
的排序。

1.2排序运用

1.3 常见的排序算法 

// 排序实现的接口
// 插入排序
void InsertSort(int* a, int n);
// 希尔排序
void ShellSort(int* a, int n);
// 选择排序
void SelectSort(int* a, int n);
// 堆排序
void AdjustDwon(int* a, int n, int root);
void HeapSort(int* a, int n);
// 冒泡排序
void BubbleSort(int* a, int n)
// 快速排序
void QuickSort(int* a, int left, int right);
// 归并排序
void MergeSort(int* a, int n)

二、插入排序

2.1基本思想: 

直接插入排序是一种简单的插入排序法,其基本思想是:把待排序的记录按其关键码值的大小逐
个插入到一个已经排好序的有序序列中,直到所有的记录插入完为止,得到一个新的有序序列 。
实际中我们玩扑克牌时,就用了插入排序的思想 。

2.2直接插入排序: 

当插入第i(i>=1)个元素时,前面的array[0],array[1],…,array[i-1]已经排好序,此时用array[i]的排
序码与array[i-1],array[i-2],…的排序码顺序进行比较,找到插入位置即将array[i]插入,原来位置
上的元素顺序后移 直接插入排序的特性总结:

1. 元素集合越接近有序,直接插入排序算法的时间效率越高
2. 时间复杂度是多少?O(N^2)
什么情况下最坏?逆序  1+2+3+...+n-1
什么情况下最好?顺序有序  O(N)
3. 空间复杂度:O(1),它是一种稳定的排序算法
4. 稳定性:稳定

2.3步骤:

  1. 确定已排序和未排序部分:
    初始时,认为数组的第一个元素(索引为0)是已排序部分,其余元素是未排序部分。
    从数组的第二个元素开始(索引为1),逐步将未排序部分的元素插入到已排序部分。
  2. 选择待插入元素:
    未排序部分中选择第一个元素作为待插入元素(开始时是第二个元素)。
    在后续的迭代中,逐步选择未排序部分的下一个元素。
  3. 查找插入位置:
    从已排序部分的最后一个元素开始,逐个与待插入元素比较。
    如果已排序部分的当前元素大于待插入元素,则将当前元素后移一位,为待插入元素腾出空间。
    如果已排序部分的当前元素小于或等于待插入元素,或者已到达已排序部分的开头(即索引为0),则停止比较。
  4. 插入元素:
    将待插入元素插入到已排序部分中找到的正确位置。
    插入位置是通过前面步骤中元素的后移操作确定的。
  5. 重复过程:
    重复步骤2到步骤4,直到未排序部分没有元素为止。
    每次迭代后,已排序部分会增加一个元素,而未排序部分会减少一个元素。

2.4直接插入排序的实现


void InsertSort(int* a, int n)
{// [0, end] 有序  把end + 1位置的值插入[0,end],让[0, end+1]有序for (int i = 0; i < n - 1; ++i)//控制前n个数字有序{int end = i;//从第i+1个数开始排序//每一次排序要经i次循环比较int tmp = a[end + 1];//tmp保存要排序的下一个数据(end 后一个值)while (end >= 0){if (a[end] > tmp){a[end + 1] = a[end];// 将该元素向后移动一位--end;// 继续向前查找插入位置}else {break;}}a[end + 1] = tmp;//保留的数比end都小,把保留的数放到end前面//保留的数比所有数小,end = -1,把保留的数放到所有数前面//这两种情况的操作一样,所以可以写在最外面}
}

三、希尔排序( 缩小增量排序 ) 

3.1希尔排序的发展历史

希尔排序,也被称为“缩小增量排序”或“Shell's Sort”,是插入排序的一种更高效的改进版本。该算法由D·L·Shell于1959年提出,并以他的名字命名。希尔排序是非稳定排序算法。

希尔排序的思想是将待排序的数组看作是一个矩阵,然后按一定的增量(步长)分组进行排序。通常,这个增量序列会从一个大的数值开始,然后逐渐减小到1。在增量逐渐减少的过程中,每组包含的关键词越来越多,当增量减至1时,整个文件恰被分成一组,算法便终止。在这个过程中,算法会对每一组使用直接插入排序算法进行排序。

希尔排序的诞生打破了人们普遍认为排序算法的时间复杂度无法突破O(N^2)的观念。尽管希尔排序在最坏的情况下时间复杂度仍然为O(N^2),但是在实际情况中,它的性能通常要好于这个时间复杂度。这使得希尔排序成为第一个突破平方量级瓶颈的排序算法。

希尔排序的提出在计算机科学领域引起了广泛的关注和研究。许多研究者开始寻找各种优秀的增量序列以证明这个算法的优越性。然而,直到现在,关于希尔排序的复杂度证明仍然是一个难题。目前被大部分认可的Hibbard增量在最糟糕的情况下可以把复杂度稳定在O(N^(3/2))左右,而猜想的平均复杂度大概在O(N^(5/4))。尽管这些猜想的平均复杂度都未被确切证明,但是值得肯定的是,希尔排序在实际应用中的性能通常要好于直接插入排序。

3.2 希尔排序的思路

希尔排序法又称缩小增量法。希尔排序法的基本思想是:先选定一个整数,把待排序文件中所有
记录分成个组,所有距离为的记录分在同一组内,并对每一组内的记录进行排序。然后,取,重
复上述分组和排序的工作。当到达=1时,所有记录在统一组内排好序。


希尔排序原理是每一对分组进行排序后,整个数据就会更接近有序,当增量缩小为1时,就是插入排序,但是现在的数组非常接近有序,移动的数据很少,所以效率非常高,所以希尔排序又叫缩小增量排序。

每次排序让数组接近有序的过程叫做预排序,最后一次插入是直接插入排序。

gap = 3的思路讲解

1、以3作为增量(gap)对数组进行分组,以下数组被分成3组,每组之间都是以3的等差数列

2、(gap = 2)此时gap缩小,以2为增量对数组进行分组,数组被分成2份,每组之间都是2的等差数列

3、对每一组进行插入排序,得到如下数组

 4、对每一组进行插入排序,得到如下数组

3.3 如何选择希尔增量

​希尔排序的分析是一个复杂的问题,它的时间是一个关于增量序列的函数,这涉及到一些数学上未能攻克的难题,所以目前为止对于希尔增量到底怎么取也没有一个最优的值,但是经过大量研究已经有一些局部的结论,在这里并不展开叙述。

​ 最初希尔提出的增量是 gap = n / 2,每一次排序完让增量减少一半gap = gap / 2,直到gap = 1时排序变成了直接插入排序。后面也有人提出的gap = [gap / 3] + 1,每次排序让增量成为原来的三分之一,加一是防止gap <= 3时gap = gap / 3 = 0的发生,导致希尔增量最后不为1。这二者思路都可以。

四、希尔排序的代码实现

直接插入排序的基础上的优化
1、先进行预排序,让数组接近有序
2、直接插入排序
时间复杂度:O(N*log{_{2}}N) 或者 O(N*log{_{3}}N)
平均的时间复杂度是O(N^1.3)

void ShellSort(int* a, int n)
{int gap = n;while (gap > 1){//gap = gap / 2;  // logNgap = gap / 3 + 1; // log3 N 以3为底数的对数// gap > 1时都是预排序  接近有序// gap == 1时就是直接插入排序 有序// gap很大时,下面预排序时间复杂度O(N)// gap很小时,数组已经很接近有序了,这时差不多也是(N)// 把间隔为gap的多组数据同时排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;}}
}

祝大家新年快乐!!!

看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注!

你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。

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

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

相关文章

子进程继承了内存映射mmap导致flock文件锁泄露

背景 想要利用文件锁在进程退出时自动释放的机制&#xff0c;让守护进程管擦某些资源的释放&#xff0c;并执行相关清理/重启工作。然而发现守护进程在父进程退出时永远饿死。 问题伪代码 主进程子进程 安装信号捕获&#xff0c;每当有信号时print并退出&#xff0c;对子进…

解决跨域问题(SpringBoot)

“什么是跨域&#xff1f;” 跨域 &#xff08;Cross-Origin&#xff09; 是指在浏览器的同源策略&#xff08;Same-Origin Policy&#xff09;下&#xff0c;一个网页的源&#xff08;指协议、域名、端口号的组合&#xff09;与另一个网页的源不同。因此&#xff0c;不同源的…

提升WiFi安全的有效措施

今天买了一个USB无线网卡&#xff0c;想知道这个黑客攻击WiFi的基本流程是怎么样的&#xff0c;买回来意外就发生了&#xff0c;哈哈&#xff1a;这个网卡只支持2GWiFi网络类型&#xff0c;并且不支持WPA3网络加密方式。 你不能希望黑客永远是一个有很多钱的人&#xff0c;或许…

如何提高提问的水平?

以下是由 Google Bard 的回答。 提高提问水平&#xff0c;可以从以下几个方面入手&#xff1a; 明确问题的目的。 提问之前&#xff0c;先明确自己想通过提问达到什么目的。是想获得某个信息&#xff1f;是想了解某个概念&#xff1f;还是想解决某个问题&#xff1f;明确目的&…

综合场景搭建、在线分享,这款地理空间数据管理软件功能太多了!

《四维轻云》是一款轻量化的地理空间数据管理云平台&#xff0c;支持地理空间数据的在线管理、编辑及分享。平台具有项目管理、数据上传、场景搭建、发布分享、SDK开发等功能模块&#xff0c;支持多用户在线协作管理&#xff0c;实现了轻量化、便捷化的空间数据应用。 一、发布…

YOLOv5改进Shape-IoU损失函数:元旦假期最新 IoU论文 | 考虑边界框形状和比例的更准确的指标,YOLO性能提升

💡本篇内容:YOLOv5改进Shape-IoU损失函数:元旦假期最新 IoU论文 | 考虑边界框形状和比例的更准确的指标,YOLO性能提升 💡🚀🚀🚀本博客 改进源代码改进 适用于 YOLOv5 按步骤操作运行改进后的代码即可 💡论文地址:https://arxiv.org/abs/2312.17663 2023年12月…

linux下超级程序!在linux界面实现类图像化界面的操作体验!

linux下超级程序&#xff01;在linux界面实现类图像化界面的操作体验&#xff01; 本期带来一个超级程序&#xff01;在linux界面实现类图像化界面的操作体验。具体功能代码如下: 1500行完整代码想要完成部署&#xff0c;只需在本地创建一个LinuxGJ.sh的文件&#xff0c;然后…

vue+ts element-plu是页码器根据屏幕宽度变化,解决刷新后初始化值问题

实现思路&#xff1a;组件挂载后执行初始化操作&#xff0c;初始化添加事件监听器&#xff0c;当浏览器窗口大小发生变化时会调用这个函数handleResize <el-pagination v-model:current-page"currentPage" background :total"total" layout"prev,…

Go中interface != nil不一定不是nil

摘要&#xff1a; interface{} 值 ! nil不一定不是nil&#xff0c;应使用reflect库判断是否是nil。 测试示例&#xff1a; // todo interface ! nil 不一定 不是nil var value map[string]interface{} reqMap : make(map[string]interface{}) reqMap["key"] valu…

Vue.js 3.4版本发布:解析速度提升2倍,双向绑定革新等新功能

引言 随着2024年的来临,Vue团队的领军人物Evan You宣布了Vue.js 3.4的发布。这个版本不仅仅是修复了一些bug,还带来了一些非常实用的新功能和性能提升。 解析速度提升2倍 这次更新中,Vue.js 3.4实现了解析速度的大幅提升。尤其是在构建模板和脚本的源代码映射时,单文件组…

优维科技2024战略定位:新一代运维核心系统提供商

01 经济复苏「走远路」 过去几年&#xff0c;全球经济持续低迷&#xff0c;2024会迎来转机吗&#xff1f; 回顾2023年&#xff0c;尽管经济复苏动能式微&#xff0c;但全球经济因有效控制通胀而展现出来的韧性&#xff0c;让包括中国在内的大部分经济体躲过了深度衰退的陷阱&…

C语言所有操作符总结

目录 算术操作符&#xff1a; 移位操作符&#xff1a; 位操作符&#xff1a; 赋值操作符&#xff1a; 单目操作符&#xff1a; 关系操作符&#xff1a; 逻辑操作符&#xff1a; 以及特殊的操作符&#xff08;条件&#xff0c;逗号&#xff0c;下标&#xff0c;调用&…

鸿蒙开发第一天

一、开发准备工作 1、开发工具的安装 1&#xff09;下载地址&#xff1a;https://developer.huawei.com/consumer/cn/deveco-studio/ 2&#xff09;查询API文档链接&#xff1a;https://developer.huawei.com/consumer/cn/doc/harmonyos-references-V2/syscap-00000014080893…

Spring之bean的实例化方式

1.使用构造方法实例化bean&#xff08;利用反射&#xff09; import lombok.Data;Data public class People {private String name;private Integer age;private String eat; }<?xml version"1.0" encoding"UTF-8"?> <beans xmlns"http:/…

【MySQL】关于日期转换的方法

力扣题 1、题目地址 1853. 转换日期格式 2、模拟表 表: Days Column NameTypedaydate day 是这个表的主键。 3、要求 给定一个Days表&#xff0c;请你编写SQL查询语句&#xff0c;将Days表中的每一个日期转化为"day_name, month_name day, year"格式的字符串…

【C语言】编程世界的不朽基石与未来展望

C语言&#xff0c;一种经久不衰的高级编程语言&#xff0c;自1972年由Dennis Ritchie在AT&T贝尔实验室开发以来&#xff0c;已深深扎根于编程语言的发展历程中。它既是计算机科学史上的一个重要里程碑&#xff0c;也是现代软件开发的核心支柱。从操作系统到嵌入式系统的构建…

一篇关于大模型在信息抽取(实体识别、关系抽取、事件抽取)的研究进展综述

信息提取&#xff08;IE&#xff09;旨在从普通自然语言文本中提取结构化知识&#xff08;如实体、关系和事件&#xff09;。最近&#xff0c;生成式大型语言模型&#xff08;LLMs&#xff09;展现了在文本理解和生成方面的卓越能力&#xff0c;使得它们能够广泛应用于各种领域…

Java解析xml文档,判断对象是一个json是jsonArray还是jsonObject

有一篇xml文档&#xff0c;如下&#xff1a; 现在需要解析出其中的内容&#xff0c;首先需要明确的是&#xff0c;文档是由一个个的标签嵌套形成的&#xff0c;例如整个xml文件是由许多DescriptorRecord标签构成&#xff0c; <DescriptorRecord DescriptorClass "1&…

基于ssm的旅游网页开发与设计+jsp论文

摘 要 信息数据从传统到当代&#xff0c;是一直在变革当中&#xff0c;突如其来的互联网让传统的信息管理看到了革命性的曙光&#xff0c;因为传统信息管理从时效性&#xff0c;还是安全性&#xff0c;还是可操作性等各个方面来讲&#xff0c;遇到了互联网时代才发现能补上自古…

linuxnodejs 20.* 安装问题,version `GLIBCXX_3.4.26‘

背景 今天服务器被重置拉&#xff0c;nodejs 环境不存在&#xff0c;特意安装下nodejs&#xff0c;一访问官网&#xff0c;妈呀&#xff0c;居然到20版本拉&#xff01;就尝试安装下最新版本&#xff01; 过程 $ cd /opt $ curl -OL https://nodejs.org/dist/v20.10.0/node-v2…