当前位置: 首页 > news >正文

数据结构之插入排序

目录

前言

插入排序

直接插入排序

插入排序的时间复杂度

希尔排序 


前言

在日常生活中,我们不经意间会遇到很多排序的场景,比如在某宝,某东上买东西,我们可以自己自定义价格是由高到低还是由低到高,再比如在王者某耀中的每个英雄的荣耀战力,都是由高到低进行排序的,这些场景都用到了排序,但是这些场景的底层都是用一个个排序算法来实现的,本期开始,我们就要学习数据结构中很重要的一个知识点-------排序。

 几种常见的排序:

注:我们今后所有的排序都默认排升序。

本期我们主要讲解插入排序。

插入排序

直接插入排序

直接插入排序在现实生活中最常见的应用其实就是斗地主时,我们一般会将牌拍好序,然后根据自己的打法打牌,大家先仔细想想,再给牌整理时我们是怎样整理的呢?当我们摸到第一张牌时,此时它已经有序,我们摸第二张牌时,就会与第一张牌做比较,然后发生交换使之成为我们想要的顺序,当我们摸第三张牌时,我们又会与最后一张牌与倒数第二张牌进行比较,交换之后使牌的顺序成为我们想要的顺序,之后每摸一张牌,就按照此方法进行整理,最后就会得到一个有序的牌列。这就直接插入排序的一个比较常见的应用场景。在数据结构中,我们怎样实现直接插入排序呢?

要实现直接插入排序,我们可以先将直接插入排序分解为单趟排序,然后再让每趟排序组合成为整个排序。

单趟排序:

单趟排序的情景:我们要将一个元素插入一个有序数组之中。可以类比,我们摸了一张牌还没有插入手中的牌中,但是手中的牌肯定已经是一个有序的牌序列了。

单趟排序的算法思想:我们令有序的数组的最后一个元素的位置为end,要插入的元素为x。要将x插入一个有序数组,就得先让x与end位置上的元素进行比较,因为我们是排升序,所以如果x比a[end]小,就要把a[end]向后挪动一个位置,然后将end--,然后在让x与此时end位置上的元素a[end]进行比较,比较的原理同上,直到将x插入到合理的位置,那么怎样判断这样的一趟排序已经结束了呢?其实就是x比当前end位置上的元素要大,这是一个结束条件,还有就是x比有序数组所有的元素都要小,即就是要将x放在数组的第一个元素的位置上,此时end已经到了数组第一个元素位置的前一个位置之前,我们称此时end==-1,所以综上,一趟排序的结束标志就是x>a[end]或者end<0

单趟排序代码实现:

int end = size - 1;int x;while (end >= 0){if (a[end] > x){a[end + 1] = a[end];end--;}else {break;}}a[end + 1] = x;

 到了这里,我们单趟排序已经搞定了,但是上面我们已经说了,我们把整个插入排序分为了每趟排序,那么怎样将每趟排序合并成为直接插入排序呢?

每趟排序的关键点在于是把一个元素插入一个有序数组之中,其实对于一个数组而言,无论它是否是有序还是无序的,其第一个元素毋庸置疑肯定是有序的,这便是关键,我们可以从第一个与元素开始,把第一个元素当成一个有序数组,此时第二个元素就是要插入有序数组中的元素,那么此时第二哥元素插入由第一个元素组成的有序数组中就可以看成是一趟排序,这趟排序之后,第一个元素和第二个元素就组成了一个有序数组,第三个元素插入由第一个和第二个元素组成的有序数组就可以看成第二趟排序,由此依次进行多趟排序,那么整个数组的多趟排序就最终组成了直接插入排序。

直接插入排序完整代码:

void InsertSort(int* a, int size)
{assert(a);for(int end=0;end<size-1;end++){ int x = a[end + 1];while (end >= 0){if (a[end] > x){a[end + 1] = a[end];end--;}else{break;}}a[end + 1] = x;}
}
int main()
{int arr[] = { 10,9,8,7,6,5,4,3,2,1 };InsertSort(arr, sizeof(arr) / sizeof(int));for (int i = 0; i < sizeof(arr) / sizeof(int); i++){printf("%d ", arr[i]);}return 0;}

运行截图如下:

注意:

排序这里比较重要的就是边界的控制,对于直接插入排序而言,若数组的大小为sizeend的初始值就是0end的最终值就是size-2x的初始位置就是1x的最终位置就是size-1。

以上便是直接插入排序的整个过程。

插入排序的时间复杂度

最好:O(N),已经有序的数组,要插入的元素比end位置上的元素大,只用跟end位置上的元素比较一下。

最坏:O(N^2),逆序,要插入的元素比有序数组的所有元素都要小,跟end位置和end位置之前的所有元素都要比较一下。

稳定性:稳定。

希尔排序 

希尔排序其实就是插入排序的进阶版,我们知道,一个数组越有序,直接插入排序的时间复杂度越低,希尔排序其实就是为了将一个无序的数组变得有序,使插入排序变得更加简单。

希尔排序如图:

希尔排序的思想:先设置一个gap值,先把整个数组的元素分成gap组,即下标差gap的为一组,然后对每一组按照直接插入排序的思想进行排序,每一组的排序将完成一次预排序,每完成一次预排序,数组会变得比之前更加有序。我们再改变gap的值多次进行预排序,当gap==1时,就是直接插入排序,此时因为经历了多次预排序,整个数组已经变得有序,所以此时再用直接插入排序对整个数组完成最终的排序。

希尔排序的整体代码:

void ShellSort(int* a, int size)
{int gap = size;while (gap > 1){gap /= 2;//进行一趟预排序for (int i = 0; i < gap; i++){//进行每组排序for (int end = i; end < size - gap; end += gap){int x = a[end + gap];//进行单趟排序while (end >= 0){if (a[end] > x){a[end + gap] = a[end];end -= gap;}else{break;}}a[end + gap] = x;}}}
}
int main()
{int arr[] = { 10,9,2,8,6,5,4,3,11,1 };ShellSort(arr, sizeof(arr) / sizeof(int));for (int i = 0; i < sizeof(arr) / sizeof(int); i++){printf("%d ", arr[i]);}return 0;
}

 运行截图如下:

当然,上述的方法是把三组的排序是分开的,每组自己排自己的,但是下面这种方法就是三种排序我们再一次排序中就全部排序完成,更为简洁,但是没有第一种好理解,可以理解为是第一种方法的进阶版本,但是这种进阶版本是我们经常使用的。

进阶版本代码如下:

void ShellSort(int* a, int size)
{int gap = size;while (gap > 1){gap /= 2;//进行一趟预排序,将多组排序在一次排序中就完成了,而不是按组进行排序for (int end = 0; end < size - gap; end++){int x = a[end + gap];//进行单趟排序while (end >= 0){if (a[end] > x){a[end + gap] = a[end];end -= gap;}else{break;}}a[end + gap] = x;}}
}
int main()
{int arr[] = { 10,19,2,8,6,0,4,3,11,1 };ShellSort(arr, sizeof(arr) / sizeof(int));for (int i = 0; i < sizeof(arr) / sizeof(int); i++){printf("%d ", arr[i]);}return 0;
}

以上便是希尔排序的整个过程。

希尔排序的时间复杂度

 希尔排序的时间复杂度为:

1.最好:O(N),当整个数组为有序数组时,复杂度最小。排序算法的时间复杂度的天花板就是O(N)

 2.最坏:总共分成了gap组,每组大概就是size/gap个元素,如果每组都逆序,那么每组的复杂度又是一个1+2+...+size/gap的等差数列,平均是O(N^1.3)。

稳定性:不稳定。

插入排序的内容就是这些,我们需要注意的仍然是算法中边界的控制,在编写代码时,我们可以自己画图去控制边界,这是一个很好的方法。

好了,本期的内容到此结束^_^ 

相关文章:

数据结构之插入排序

目录 前言 插入排序 直接插入排序 插入排序的时间复杂度 希尔排序 前言 在日常生活中&#xff0c;我们不经意间会遇到很多排序的场景&#xff0c;比如在某宝&#xff0c;某东上买东西&#xff0c;我们可以自己自定义价格是由高到低还是由低到高&#xff0c;再比如在王者某…...

2023年江西省“振兴杯”网络信息行业(信息安全测试员)职业技能竞赛 Write UP

文章目录 一、2023csy-web1二、2023csy-web2三、2023csy-web3四、2023csy-web4五、2023csy-misc1六、2023csy-misc2七、2023csy-crypto1八、2023csy-re1 一、2023csy-web1 该题提供一个web靶场&#xff0c;《伟大的挑战者》&#xff0c;分值&#xff1a;5分 web页面一直在播放c…...

【5G PHY】5G NR 如何计算资源块的数量?

博主未授权任何人或组织机构转载博主任何原创文章&#xff0c;感谢各位对原创的支持&#xff01; 博主链接 本人就职于国际知名终端厂商&#xff0c;负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作&#xff0c;目前牵头6G算力网络技术标准研究。 博客…...

解决oracle.sql.TIMESTAMP序列化转换失败问题 及 J2EE13Compliant原理

目录 报错现象报错内容处理方法Oracle驱动源码总结 报错现象 oracle表中存在TIMESTAMP类型的列时&#xff0c;jdbc查出来做序列化时报错 报错内容 org.springframework.web.util.NestedServletException: Request processing failed; nested exception is org.springframewo…...

QQ2023备份

需要修改的路径&#xff08;共3处&#xff09; 这三处路径中&#xff0c;只有一处是需要修改的 QQPC端-主菜单-设置-基本设置-文件管理 点击上面的“”自定义“”&#xff0c;然后修改路径即可 修改路径后提示 然后等一会才会关干净QQ的相关进程&#xff0c;关闭后才会有自动…...

HNU计算机结构体系-实验2:CPU动态指令调度Tomasulo

文章目录 实验2 CPU动态指令调度Tomasulo一、实验目的二、实验说明三、实验内容问题1&#xff1a;问题2&#xff1a;问题3&#xff1a;问题4&#xff1a;问题5&#xff1a; 四、思考题问题1&#xff1a;问题2&#xff1a; 五、实验总结 实验2 CPU动态指令调度Tomasulo 一、实验…...

智慧城市是什么?为什么要建智慧城市?

智慧城市是一个通过现代科技手段推动城市管理和服务创新的概念。 具体来说&#xff0c;它利用信息技术和创新概念&#xff0c;将城市的各个系统和服务集成起来&#xff0c;以提升城市运行效率、优化城市管理和服务&#xff0c;改善市民的生活质量。 为什么要建智慧城市呢&…...

数据结构线性表-栈和队列的实现

1. 栈(Stack) 1.1 概念 栈&#xff1a;一种特殊的线性表&#xff0c;其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈 顶&#xff0c;另一端称为栈底。栈中的数据元素遵守后进先出LIFO&#xff08;Last In First Out&#xff09;的原则。 …...

IntelliJ IDEA 的 HTTP 客户端的高级用法

本心、输入输出、结果 文章目录 IntelliJ IDEA 的 HTTP 客户端的高级用法前言HTTP 请求对 gRPC 请求的支持对 GraphQL 和 WebSocket 请求的支持环境文件OpenAPI 补全用于持续集成的 HTTP 客户端 CLI花有重开日,人无再少年实践是检验真理的唯一标准IntelliJ IDEA 的 HTTP 客户端…...

代码随想录算法训练营第四十六天 _ 动态规划_198.打家劫舍、213.打家劫舍II、337.打家劫舍 III。

学习目标&#xff1a; 动态规划五部曲&#xff1a; ① 确定dp[i]的含义 ② 求递推公式 ③ dp数组如何初始化 ④ 确定遍历顺序 ⑤ 打印递归数组 ---- 调试 引用自代码随想录&#xff01; 60天训练营打卡计划&#xff01; 学习内容&#xff1a; 198.打家劫舍 动态规划五步曲&a…...

ffmpeg编译问题

利用ffmpeg实现一个播放器&#xff0c;ffmpeg提供动态库&#xff0c;但是编译链接的时候遇到下面的问题&#xff1a; ../ffmpegWidgetPlayer/videoplayerwidget.cpp:23: error: undefined reference to sws_freeContext(SwsContext*) ../ffmpegWidgetPlayer/videoplayerwidget.…...

【flink番外篇】1、flink的23种常用算子介绍及详细示例(3)-window、distinct、join等

Flink 系列文章 一、Flink 专栏 Flink 专栏系统介绍某一知识点&#xff0c;并辅以具体的示例进行说明。 1、Flink 部署系列 本部分介绍Flink的部署、配置相关基础内容。 2、Flink基础系列 本部分介绍Flink 的基础部分&#xff0c;比如术语、架构、编程模型、编程指南、基本的…...

centos7做gitlab数据灾备项目地址指向问题

如果你在 CentOS 7 上使用 GitLab 时&#xff0c;它回复的数据指向了另一个服务器的地址&#xff0c;可能是因为配置文件中的一些设置不正确。 要解决这个问题&#xff0c;可以尝试以下几个步骤&#xff1a; 检查 GitLab 配置文件&#xff1a;打开 GitLab 的配置文件&#xf…...

leetcode:93. 复原 IP 地址

复原 IP 地址 中等 1.4K 相关企业 有效 IP 地址 正好由四个整数&#xff08;每个整数位于 0 到 255 之间组成&#xff0c;且不能含有前导 0&#xff09;&#xff0c;整数之间用 ‘.’ 分隔。 例如&#xff1a;“0.1.2.201” 和 “192.168.1.1” 是 有效 IP 地址&#xff0c;但…...

玄子Share-CSS3 弹性布局知识手册

玄子Share-CSS3 弹性布局知识手册 Flexbox Layout&#xff08;弹性盒布局&#xff09;是一种在 CSS 中用于设计复杂布局结构的模型。它提供了更加高效、简便的方式来对容器内的子元素进行排列、对齐和分布 主轴和交叉轴 使用弹性布局&#xff0c;最重要的一个概念就是主轴与…...

Nat easy IP ACL

0表示匹配&#xff0c;1表示任意&#xff08;主机位0.0.0.255&#xff08;255主机位&#xff09;&#xff09; rule deny source 192.168.2.1 0 设置拒绝192.168.2.1的主机通过 记住将其应用到接口上 [AR2]acl 2000 //创建基本ACL [AR2-acl-basic-2000]rule deny source 192…...

Numpy数组的数据类型汇总 (第4讲)

Numpy数组的数据类型 (第4讲)         🍹博主 侯小啾 感谢您的支持与信赖。☀️ 🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ�…...

通讯app:

为了开发一个即时通讯的app&#xff0c;包含发送文字、语音、视频以及视频通话的功能&#xff0c;我们需要考虑以下的技术栈和实现步骤&#xff1a; 技术栈建议&#xff1a; 前端&#xff1a;React Native 或 Flutter 用于跨平台移动应用开发。后端&#xff1a;ThinkPHP Wor…...

【Backbone】TransNeXt:最新ViT模型(原理+常用神经网络汇总)

文章目录 一、近几年神经网络 Backbone 回顾1.Densenet 与 Resnet2.CBP3.SENet4.GCNet5.DANet6.PANet 与 FPN7.ASPP8.SPP-net9.PSP-net10.ECA-Net 二、TransNeXt&#xff08;2023&#xff09;1.提出问题2.Aggregated Pixel-focused Attention2.1 Pixel-focused Attention&#…...

使用Java将图片添加到Excel的几种方式

1、超链接 使用POI&#xff0c;依赖如下 <dependency><groupId>org.apache.poi</groupId><artifactId>poi</artifactId><version>4.1.2</version></dependency>Java代码如下,运行该程序它会在桌面创建ImageLinks.xlsx文件。 …...

计算机组成原理实验避坑指南:存储器地址映射常见错误及解决方法

计算机组成原理实验避坑指南&#xff1a;存储器地址映射常见错误及解决方法 第一次在Proteus里搭建存储器系统时&#xff0c;看着密密麻麻的地址线和片选信号&#xff0c;我对着实验指导书发呆了半小时——明明按照图示连接了所有线路&#xff0c;可写入RAM的数据总是莫名其妙出…...

LTI系统设计避坑指南:因果性与稳定性在实际工程中的5个关键检查点

LTI系统设计避坑指南&#xff1a;因果性与稳定性在实际工程中的5个关键检查点 在数字信号处理领域&#xff0c;线性时不变&#xff08;LTI&#xff09;系统的设计是工程师日常工作的核心。然而&#xff0c;理论推导与工程实践之间往往存在一道鸿沟——许多在数学上完美的系统模…...

python-flask-djangol框架的食品仓库管理系统

目录需求分析与功能规划技术栈选择系统架构设计开发与测试流程安全与性能优化部署方案项目技术支持源码获取详细视频演示 &#xff1a;文章底部获取博主联系方式&#xff01;同行可合作需求分析与功能规划 明确食品仓库管理系统的核心需求&#xff0c;包括库存管理、食品分类、…...

PFC颗粒流代码模拟岩石预制裂隙与完整岩石单轴压缩对比分析

PFC颗粒流代码 pfc离散元岩石预制裂隙&#xff0c;裂隙岩石与完整岩石单轴压缩代码&#xff0c;可出各种裂隙形式&#xff0c;可分析应力应变曲线图&#xff0c;裂隙发育与数量&#xff0c;能量变化&#xff0c;简易声发射分析等做岩石单轴压缩离散元模拟的&#xff0c;谁没为…...

通义千问3-Reranker-0.6B优化升级:调整批处理大小和自定义指令,性能再提升5%

通义千问3-Reranker-0.6B优化升级&#xff1a;调整批处理大小和自定义指令&#xff0c;性能再提升5% 1. 为什么需要优化重排序模型性能&#xff1f; 在信息检索和问答系统中&#xff0c;重排序模型扮演着至关重要的角色。它负责对初步检索得到的文档进行二次排序&#xff0c;…...

如何快速找到领域内的核心论文?3 条最有效路径

在做科研文献检索时&#xff0c;很多研究者都会遇到同一个问题&#xff1a; 文献很多&#xff0c;但不知道哪些最重要。例如&#xff0c;当你在数据库中输入一个研究关键词时&#xff0c;检索结果可能会出现几百篇甚至上千篇论文。面对如此庞大的文献数量&#xff0c;很多人会产…...

tkinter表格神器tkintertable实战:5分钟搞定可拖拽编辑的数据表格(附完整代码)

tkinter表格神器tkintertable实战&#xff1a;5分钟搞定可拖拽编辑的数据表格&#xff08;附完整代码&#xff09; 在Python GUI开发中&#xff0c;表格控件一直是刚需但实现起来又颇为棘手的组件。传统tkinter自带的Treeview虽然能勉强实现表格功能&#xff0c;但在交互体验上…...

CANOE Demo版快速下载与激活指南

1. CANOE Demo版是什么&#xff1f;为什么你需要它&#xff1f; 如果你正在学习汽车电子开发或者从事相关领域的工作&#xff0c;CANOE这个名字一定不会陌生。作为Vector公司推出的主流汽车总线开发工具&#xff0c;它几乎成了行业标准。但对于刚入门的新手来说&#xff0c;动辄…...

2026年江苏省职业院校技能大赛(学生组)信息安全管理与评估(技能操作阶段)竞赛样题

2026年江苏省职业院校技能大赛&#xff08;学生组&#xff09;信息安全管理与评估&#xff08;技能操作阶段&#xff09;竞赛样题 文章目录2026年江苏省职业院校技能大赛&#xff08;学生组&#xff09;信息安全管理与评估&#xff08;技能操作阶段&#xff09;竞赛样题任务1&a…...

EVA-01场景应用:电商商品分析、文档信息提取,真实工作流分享

EVA-01场景应用&#xff1a;电商商品分析、文档信息提取&#xff0c;真实工作流分享 1. 从科幻到现实&#xff1a;EVA-01的商业价值 在电商运营和文档处理的日常工作中&#xff0c;我们常常面临这样的挑战&#xff1a;海量商品图片需要人工标注关键信息&#xff0c;繁杂的合同…...