直接插入排序和希尔排序

前言

我们前面几期介绍了线性和非线性的基本数据结构。例如顺序表、链表、栈和队列、二叉树等~!本期和接下来的几期我们来详解介绍各个排序的概念、实现以及性能分析!

本期内容

排序的概念以及其运用

常见的排序算法

直接插入排序

希尔排序

一、排序的概念及其运用

排序的概念

排序:按照一定的规则,把一组元素序列以递增或递减排列起来的操作!

稳定性:假设在待排序的一组序列中有多个相同的元素,若经过排序,这些元素的相对位置(相对次序)是不变的,则称为这种排序是稳定的。否则就是不稳定的!

内部排序:数据元素全部放在内存中的排序

外部排序:数据元素太多不能同时在内存中,根据排序的过程要求不能在内存和外存之间移动数据的排序!

关于哪些是内部排序、哪些是外部排序后面性能分析会在介绍!

排序的运用

排序在日常生活中是极其常见的:例如你每天看的京东、淘宝、拼夕夕等购物平台上的综合、销量、好评、价格、等一系列都是对商品的排序!

还有高校排名:

还有大家考试的时候的排名,这些都是排序~!排序在生活只能是很常见的!!!

常见的排序算法

二、直接插入排序及其实现

插入排序的基本思想

待排序的元素序列按照大小逐个插入到已经排序好的有序序列中,直到所有待排序的元素插入完为止, 而此时得到的新的序列就是有序的序列!

这就和我们小时候玩扑克牌摸牌整理的一样,一次与前面的排比较找到合适的位置插入!

直接插入排序

第i个元素插入时(i >= 1)前面的序列已经有序的, 此时只需要用array[i]的元素与前面的所有元素逐一比较前面的元素比当前的前面的元素插入到当前位置(升序),否则当前元素插入到前面元素的后面!

我们根据上述思路写单趟改造整体

	int end = 0;//一开始end置0位置int tmp = a[end + 1];//tmp是end 的下一个位置的元素while (end >= 0){if (a[end] > tmp)//前面元素比当前的元素大{a[end + 1] = a[end];//前面元素的插入到前面元素的后一个位置}else//前面元素不比当前的元素大{a[end + 1] = tmp;//当前元素插入到前面元素的下一个位置break;//记得结束否则会又把排好的区间搞乱}--end;}if (end < 0)//所有的都比tmp大,此时end一直减会减到-1{a[0] = tmp;//此时把tmp(end的下一个位置的元素)插入到0下标位置}

整体:

其实单趟写出来,改造整体就会很容易,我们在外面在套一层循环即可!让end从i 开始,每个元素与前面的逐一比较走单趟,直至最后有序!

void InsertSort(int* a, int n)
{for (int i = 0; i < n - 1; i++){int end = i;//一开始end置0位置int tmp = a[end + 1];//tmp是end 的下一个位置的元素while (end >= 0){if (a[end] > tmp)//前面元素比当前的元素大{a[end + 1] = a[end];//前面元素的插入到前面元素的后一个位置}else//前面元素不比当前的元素大{a[end + 1] = tmp;//当前元素插入到前面元素的下一个位置break;//记得结束否则会又把排好的区间搞乱}--end;}if (end < 0)//所有的都比tmp大,此时end一直减会减到-1{a[0] = tmp;//此时把tmp(end的下一个位置的元素)插入到0下标位置}}
}

但要注意的是:外面的for循环的判断条件,i < n - 1, 也就是说i最多走到n - 2的位置即倒数第二个元素!原因是:tmp才是每次要插入的元素,而tmp = a[end +1]是end(i)的下一个位置,如果让i 到最后一个元素的位置即n-1处,那tmp = a[end+1]就会越界!!!所以i 只能到倒数第二个元素的位置!

OK, 直接插入写完了,测试一下:

这样写是没问题,但感觉稍微有点挫~!我们在当前元素不大于前面元素的时候要判断一次,是否越界也要判断一次。

我们能不能想办法给优化一下呢?答案是肯定的!我们无论是判断当前的元素不比前面的大还是越界最后插入的都是end+1的位置,所以当在单趟的循环(while)内一旦不满足当前的比前面的大则立刻跳出当前循环。到了单趟循环外有可能是break结束的,也有可能是end < 0结束的,但我们根本不需要关心他,直接插入到end+1的位置即可~!

void InsertSort(int* a, int n)
{for (int i = 0; i < n - 1; i++){int end = i;int tmp = a[end + 1];while (end >= 0){if (a[end] > tmp){a[end + 1] = a[end];}else{break;}--end;}a[end + 1] = tmp;}
}

这样写简洁了不少吗,也不容易出错了~!我以前学的时候老是把上面第一种的break给忘了,最后找半天(Q^Q)....所以建议大家一般写第二种~!

复杂度分析

时间复杂度:O(N^2)  ---> 单趟是O(N),最坏情况N个元素都要走一次单趟

空间复杂度:O(1)  ---> 额外使用空间的个数是常数个

当要排序的序列接近有序时性能最好O(N)~!

OK, 我们直接插入排序就介绍到这里,下面我们来介绍一下它的优化版 ----- 希尔排序!!

三、希尔排序及其实现

我们上面介绍过直接插入的时间复杂度是O(N^2),它的性能一般~!有一天一位叫D.L.Shell的大佬去学习了直接插入排序后,想既然你直接插入排序在接近有序的情况下性能很好(O(N)),那我能不能把一组无序的元素经过处理先让他变得接近有序然后再来一趟直接插入呢?其实他这种想法直接把直接插入排序优化到几乎能和快排平起平坐了~!就是下面这位大佬:

OK,我们来看看大佬的具体思路!

希尔排序的思路

1、进行多组预排序

2、最后再来一次直接插入

什么意思呢?我来解释一下:这里的多组预排是:

先选定一个整数增量gap(一开始gap = n / 2),将数组的元素分为gap 组,将每个距离为gap的分为一组,并对每个小组进行距离为gap的"直接插排",然后不断的缩小gap(gap /= 2),重复上述操作,直到gap == 1时就说明各个组的都已经排好,此时相较于一开始已经是非常有序的了~!最后我们再来一趟直接插入排序则该序列就是有序的了!

OK, 画个图理解一下:

这就将一个完整的数组分成了gap 组,此时的gap 是 5,我们走一下预排序的一个gap:

他这样走下去,gap最后一定会到gap == 1,当gap == 1那就是直接插排了:

上述栗子可以清楚的看到当gap == 1时,未开始最后一次直接插排前的数组已经是很有序了,当经过最后一趟直接插入排序后就是有序了~!

希尔排序的实现

还是先单趟,再整体~!

单趟

每一组的单趟

for (int i = 0; i < n - gap; i += gap)
{int end = i;int tmp = a[end + gap];while (end >= 0){if (a[end] > tmp){a[end + gap] = a[end];}else{break;}end -= gap;}a[end + gap] = tmp;
}

注意的是这里是n - gap 而不是n - gap - 1

这是一组的单趟,这里和直接插入排序几乎一样,当gap == 1就是直接插排~!此时有多少组,就走多少组这样的单趟,所以在外面在套一层循环才是所有组的单趟~!

所有组的单趟

for (int i = 0; i < gap; i++)
{for (int i = 0; i < n - gap; i += gap){int end = i;int tmp = a[end + gap];while (end >= 0){if (a[end] > tmp){a[end + gap] = a[end];}else{break;}end -= gap;}a[end + gap] = tmp;}
}

这就是所有的预排序的单趟,那他到底走多少趟预排呢?具体不知道,当gap最后通过调整到(gap /= 2或gap = gap / 3 + 1) gap == 1即可!所以总体应该在外面套个循环:

void ShellSort(int* a, int n)
{int gap = n;while (gap > 1){gap /= 2;for (int i = 0; i < gap; i++){for (int i = 0; i < n - gap; i += gap){int end = i;int tmp = a[end + gap];while (end >= 0){if (a[end] > tmp){a[end + gap] = a[end];}else{break;}end -= gap;}a[end + gap] = tmp;}}}
}

gap 一开始是n,然后开始执行第一趟时时gap /= 2;然后每次调整都是/=2,最后一次进入循环一定是1(n, n / 2, n / 4, n / 8, n / 16, n /32 ..... 4, 2, 1),即直接插入排序了~!

OK,测试一下:

没问题!但这段代码被另一位大佬进行过优化,如下:

void ShellSort(int* a, int n)
{int gap = n;while (gap > 1){gap = gap / 3 + 1;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];}else{break;}end -= gap;}a[end + gap] = tmp;}}
}

他这里把,单组逐一排改成了多组并排~~以前我们是一组跑完了再去排另一组,他改完后直接一遍就可以把多组排好~!但性能上没有差别,,。

这里他给的gap = gap / 3 + 1;这样写的效率其实比gap /= 2的要好一点(网上以前看到的),他这里+1是因为,gap / 3有时候会小于1(例如: 2 / 3),这样最后一趟就排不了了~!+1就可以解决这个问题!

复杂度分析

希尔排序的时间复杂度是极其不好计算的,因为它的gap 的取值方法太多,很难精确地去计算,在好多书中给的都不同:例如严蔚敏老师的书中是如下

殷人昆老师的书中如下:

由于这里我们的gap是按殷人昆老师提出的方式取的,而殷人昆老师也对其进行了大量的数据实验统计,我们可以暂时认为

希尔排序的时间复杂度为:O(N^1.25)或O(N^1.3)

希尔排序的空间复O(1)

OK,本期插入排序就先介绍到这里,好兄弟,我们下期选择排序见~!

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

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

相关文章

vue2项目从0搭建(三):配置环境变量及对应的webpack配置

前言 实际业务开发中,一个项目很可能会同时配置好几套环境。 比如:常规开发环境,开发测试环境,正式的测试环境,预发测试环境,客户甲的生产环境,客户乙的生产环境,通用生产环境,独立应用环境,微前端环境,大屏专用环境,移动端环境。 一女多嫁的实际业务场景,就需要我们进行多样…

安卓用SQLite数据库存储数据

什么是SQLite&#xff1f; SQLite是安卓中的轻量级内置数据库&#xff0c;不需要设置用户名和密码就可以使用。资源占用较少&#xff0c;运算速度也比较快。 SQLite支持&#xff1a;null&#xff08;空&#xff09;、integer&#xff08;整形&#xff09;、real&#xff08;小…

第29期 | GPTSecurity周报

GPTSecurity是一个涵盖了前沿学术研究和实践经验分享的社区&#xff0c;集成了生成预训练Transformer&#xff08;GPT&#xff09;、人工智能生成内容&#xff08;AIGC&#xff09;以及大型语言模型&#xff08;LLM&#xff09;等安全领域应用的知识。在这里&#xff0c;您可以…

Linux篇:文件系统

一、共识原理&#xff1a; 文件文件内容文件属性 磁盘上存储文件存文件的内容&#xff08;数据块&#xff09;存文件的属性&#xff08;inode&#xff09; Linux的文件在磁盘中存储是将属性和内容分开存储的。 二、硬件简述&#xff1a; 1. 认识硬件 磁盘&#xff1a;唯一的一…

混合云案例:利用 Databend Cloud 高效加速私有 Databend 的策略与实施

背景 Databend 是一款基于对象存储的存算分离湖仓产品&#xff0c;已成为云上大数据分析中高效且低成本的首选解决方案。目前&#xff0c;Databend 在多个用户场景中得到广泛应用&#xff0c;包括&#xff1a; 新媒体行业数据分析及大屏数据展示云上 CDH 替代以减少本地磁盘和…

Deep Image Prior

深度图像先验 论文链接&#xff1a;https://sites.skoltech.ru/app/data/uploads/sites/25/2018/04/deep_image_prior.pdf 项目链接&#xff1a;https://github.com/DmitryUlyanov/deep-image-prior Abstract 深度卷积网络已经成为一种流行的图像生成和恢复工具。一般来说&a…

web:[ZJCTF 2019]NiZhuanSiWei1

题目 点进题目&#xff0c;网页显示如下&#xff0c;需要代码审计 $_GET["text"]和$_GET["file"]来获取传入的两个参数text和file。使用isset()函数来检查$text变量是否已设置并且不为null。如果设置了并且不为null&#xff0c;则执行下面的逻辑。在下面的…

汽车电子 -- 车载ADAS之LCA(变道辅助系统)

相关法规文件: LCA: ISO 17387-2008 Intelligent transport systems — Lane change decision aid systems 一、变道辅助系统 LCA &#xff08;Lane Change Assist&#xff09; LCA 系统&#xff08;变道辅助系统&#xff09;监测后方相邻车道区域&#xff0c;如果有车辆在后…

《融合SCADA系统数据的天然气管道泄漏多源感知技术研究》误报数据识别模型开发

数据处理不作表述。因为我用的是处理后的数据&#xff0c;数据点这。 文章目录 工作内容1CC040VFD电流VFD转速压缩机转速反馈进出口差压 紧急截断阀开到位进出电动阀开到位发球筒电筒阀开到位收球筒电动阀开到位电动阀2005开到位越站阀开到位 工作内容2工作内容3 工作内容1 任…

Java小游戏 王者荣耀

GameFrame类 所需图片&#xff1a; package 王者荣耀;import java.awt.*; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.KeyAdapter; import java.awt.event.KeyEvent; import java.io.File; import java.util.ArrayLis…

<JavaDS> 二叉树遍历各种遍历方式的代码实现 -- 前序、中序、后序、层序遍历

目录 有以下二叉树&#xff1a; 一、递归 1.1 前序遍历-递归 1.2 中序遍历-递归 1.3 后序遍历-递归 二、递归--使用链表 2.1 前序遍历-递归-返回链表 2.2 中序遍历-递归-返回链表 2.3 后序遍历-递归-返回链表 三、迭代--使用栈 3.1 前序遍历-迭代-使用栈 3.2 中序遍…

【Python3】【力扣题】367. 有效的完全平方数

【力扣题】题目描述&#xff1a; 【Python3】代码&#xff1a; 1、解题思路&#xff1a;Python函数。num的平方根 或者 num的0.5次幂。 知识点&#xff1a;float.is_integer(...)&#xff1a;判断浮点数的值是否等于整数。也可以&#xff1a;浮点数.is_integer()。 pow(a,b)&…

【SpringCloud】微服务的扩展性及其与 SOA 的区别

一、微服务的扩展性 由上一篇文章&#xff08;没看过的可点击传送阅读&#xff09;可知&#xff0c; 微服务具有极强的可扩展性&#xff0c;这些扩展性包含以下几个方面&#xff1a; 性能可扩展&#xff1a;性能无法完全实现线性扩展&#xff0c;但要尽量使用具有并发性和异步…

【Intel FPGA】D5005 使用笔记

项目总目标&#xff0c;在AFU中实现xx算法DDR 1.FPGA device &#xff1a;1SX280HN2F43E2VG 2 .硬件架构图 3.DDR信息 4.FIM &#xff08;FPAG Interface Manager&#xff09; The FIM contains the FPGA logic to support the accelerators, including the PCIe IP core, …

UDS 相关时间参数

文章目录 UDS 全部时间参数UDS 应用层诊断时间参数1、P2 Client P2 Server P2* Client P2* Server 图例2、S3 Client S3 Server 图例 UDS CNA-TP网络层时间参数1、N_As/N_Ar 图例2、N_Bs 图例3、 N_Br 图例4、N_Cs 图例N_Cr 图例 UDS 网络层流控制时间参数 UDS 全部时间参数 UD…

Java17(LTS Long Term Support)特性

支持JDK17的主流技术框架 spring framework 6.xspringboot 3.xkafka 3.0(不在支持jdk8)jenkins 2.357&#xff08;必须jdk11起步&#xff09;James Gosling表示赶紧弃用Java8&#xff0c;使用性能最好的JDK17Chart GPT也推荐JDK17&#xff0c;从长期到性能来说。 JDK17的特性 …

【古月居《ros入门21讲》学习笔记】15_ROS中的坐标系管理系统

目录 说明&#xff1a; 1. 机器人中的坐标变换 tf功能包能干什么&#xff1f; tf坐标变换如何实现 2. 小海龟跟随实验 安装 ros-melodic-turtle-tf 实验命令 运行效果 说明&#xff1a; 1. 本系列学习笔记基于B站&#xff1a;古月居《ROS入门21讲》课程&#xff0c;且使…

数据治理框架和成熟度模型

数据治理成熟度模型 一个企业的数据治理能力越高&#xff0c;所享受到数据治理带来的价值也会越多&#xff0c;如增加收入、减少成本、降低风险等。于是&#xff0c;很多企业想要准确地评估本公司的数据治理能力&#xff0c;可以利用数据治理成熟度模型方法&#xff0c;包括 D…

求和(打表题)

题目 打个表发现当 n 时答案为 p &#xff0c;否则为 1 &#xff0c;然后套板子。 #include <iostream> #include <algorithm> #include <vector> #include <cstring> #include <cmath>using namespace std;#define int long long using i64 …

直线(蓝桥杯)

直线 题目描述 本题为填空题&#xff0c;只需要算出结果后&#xff0c;在代码中使用输出语句将所填结果输出即可。 在平面直角坐标系中&#xff0c;两点可以确定一条直线。如果有多点在一条直线上&#xff0c; 那么这些点中任意两点确定的直线是同一条。 给定平面上 2 3 个…