数据结构——直接插入排序与希尔排序(图示+文字详解)

news2024/9/17 8:18:14

内容包括:排序的代码实现,排序原理详解,代码详解,图示

part 1:直接插入排序

代码实现:

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

效果图例:

 

 排序原理:

基本原理:向一个有序区间插入一个值(此值在有序区间末尾的后一个位置)

生动些:可以将这种插入排序看作摸牌的过程,我们手上的牌都是已经按照一定顺序排列好的,现在我们又新拿了一张牌,准备插入,方法就是从手上排好的最后一张牌开始,依次比较

若是当前手上的牌大于待插入的新牌,那么需要将手上的此牌向后移动一步,腾出一个位置

若是当前手上的牌小于待插入的新牌,则新牌的位置则定下,即手上的此牌的后一个位置

代码详解:

void InsertSort(int* a, int n)
{
	int i = 0;
	for (i = 0; i < n - 1; i++)//所有有序区间的末尾位置
	{
		int end = i;//有序区间的末尾下标
		int tmp = a[i + 1];//待插入的元素先保存到tmp中
		while (end >= 0)//依次与有序区间中的所有元素一一比较
		{
			if (a[end] > tmp)
			{
				a[end + 1] = a[end];//有序区间的某个元素后移
				end--;//继续与有序区间中的元素比较
			}
			else
			{
				break;
			}
		}
		a[end + 1] = tmp;//待插入元素插入在有序区间某个元素的后面 or 越过整个有序区间,成为第一个
	}
}

1 for循环构造出所有有序区间末尾下标的可能

2 待插入元素需要临时保存在tmp变量中,因为在插入过程中伴随着有序区间元素的后移,若是不提前保存待插入元素的值,则可能导致待插入元素被覆盖

3 若是待插入元素比有序区间的元素小,则此有序区间的元素即其后面都需要后移一位,但是比较还在继续,直至待插入元素比有序区间的元素大,才会停止比较

4 停止比较的两种可能(待插入元素确定位置的两种情况):

                            1 待插入元素在有序区间内找到一个比小的元素,则它需要插在此元素的后面,即end+1的位置

                            2 待插入元素比所有有序区间内的元素都要小,此时end=-1,比无可比,则所有有序区间内的元素都后移一位,它插在数组的第一个位置上,即end+1的位置

故而在代码中我设计:若是遇到比待插入元素小的元素直接break,这样可以统一。

part 2:希尔排序

希尔排序是直接插入排序基础之上的代码设计

代码实现:

void ShellSort(int* a, int n)
{
	int gap = n;
	while (gap > 1)
	{
		gap = gap / 3 + 1;
		int i = 0;
		for (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;
		}
	}
}

排序原理:

基本原理:第一步:多次预排序使得序列逼近有序(gap>1)

                  第二步:直接插入排序,使得序列最终有序(gap==1)

间隔gap分为一组,共又gap组

gap=gap/3+1 或者gap=gap/2   无论哪种都会使得gap最终=1,就是所有元素构成一组,此时对一组的排序就是前面所介绍的直接插入排序,会让数组最终有序

代码详解:

void ShellSort(int* a, int n)
{
	int gap = n;
	while (gap > 1)//多次预排,gap=1时数组已经最终有序,结束循环
	{
		gap = gap / 3 + 1;//gap的变化,保证了gap的最后一次值是1
		int i = 0;
		for (i = 0; i < n - gap; i++)//有序区间末尾的下标,多组并排,i的变化即end的变化,end的值代表当前插入排序的是哪一组,end的取值必须<n-gap,下面有详解原因
		{
			int end = i;
			int tmp = a[end + gap];
			while (end >= 0)//对一组的某个待排序元素进行插入排序
			{
				if (a[end] > tmp)
				{
					a[end + gap] = a[end];
					end -= gap;//间隔gap为一组,所以end的前一个元素是end-gap
				}
				else
				{
					break;
				}
			}
			a[end + gap] = tmp;
		}
	}
}

下面省略每组的每个待排序元素插入自己所在组有序区间的结果,读者可自行操作,这里介绍方法 

 

 end处在红组:对红组的一个待排序元素进行插入排序,初始有序区间内只有一个9

end++,end处在绿组,对绿组的一个待排序元素进行插入排序,初始有序区间内只有一个8

end++,end处在蓝组, 对蓝组的一个待排序元素进行插入排序,初始有序区间内只有一个7

 

end++,end处在橙组, 对橙组的一个待排序元素进行插入排序,初始有序区间内只有一个6

之后的end继续++,先是将红组中的下一个待插入元素进行插入排序,再是对绿组的下一个待插入元素进行插入排序,然后是对蓝组的下一个待插入元素进行插入排序,再是对橙组的下一个待插入元素进行插入排序,红,绿,蓝,橙……如此循环多组并排

但是end最终只能抵达n-gap-1的位置,即,end<n-gap,因为我们的操作是将end+gap位置的待插入元素进行插入,必须保持数组的下标不越界,当end处在n-gap位置时,它的后一个元素end+gap就是n了,属于非法下标 ,而且当end抵达最终位置时,这一趟的预排序已经处理完成

 完结撒花~

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

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

相关文章

【已解决】ubuntu下谷歌浏览器不能上网(而火狐可以上网)

现象&#xff1a;ubuntu18.04安装的谷歌浏览器chrome不能上网(谷歌和百度都不可以)&#xff0c;而火狐可以访问谷歌和百度。 问题解决[参考]&#xff1a;将HTTPS Proxy的代理端口号也设为7890就ok了

thinkphp5---安装到宝塔出现Warning: require(): open_basedir错误

centos系统&#xff0c;nginxphp7.3 使用thinkphp5安装到宝塔的linux上&#xff0c;出现以下错误&#xff1a; Warning: require(): open_basedir restriction in effect. File(/www/wwwroot/c 解决办法&#xff1a;找到 php.ini 打开 open_basedir 修改为&#xff1a; /www/…

4.原子操作类:AtomicLong、LongAdderLong、Accumulator

JUC包中有AtomicInteger、AtomicLong和AtomicBoolean等原子性操作类&#xff0c;它们原理类似&#xff0c;下面以AtomicLong为例进行讲解。 AtomicLong 底层的操作自增自减都用Unsafe类中的getAndAddLong方法&#xff08;获取本类内存偏移值&#xff09;实现的&#xff0c;get…

UE4/5数字人Metahuman与Style3D的使用【二、布料模拟】

目录 鼠标点击布料模拟&#xff1a; 让布料模拟可以跟着动画序列&#xff1a; 有穿模情况&#xff1a; 多件衣服替换&#xff1a; 关卡序列中使用缓存&#xff1a; 效果&#xff1a; UE4/5数字人Metahuman与Style3D的使用【一、Style3DAtelier软件制作smd格式衣服并导入ue】…

Apikit 自学日记:保存、使用测试用例

API测试用例是SaaS版本企业版才能使用的功能&#xff0c;免费版用户可通过付费升级后使用。 API管理应用中的测试用例管理涉及到两个场景&#xff1a;单接口测试用例管理 和 多接口测试用例批量测试。 一、单接口测试用例管理 功能入口&#xff1a;API管理应用 / 选中某个项目…

基于Java+SSM+Vue的高校校园点餐系统设计与实现

博主介绍&#xff1a; 大家好&#xff0c;我是一名在Java圈混迹十余年的程序员&#xff0c;精通Java编程语言&#xff0c;同时也熟练掌握微信小程序、Python和Android等技术&#xff0c;能够为大家提供全方位的技术支持和交流。 我擅长在JavaWeb、SSH、SSM、SpringBoot等框架…

lesson 8下 Zigbee单播通信理论相关概念原理(端点、簇)

目录 Zigbee单播通信理论相关概念原理 端点&#xff08;Endpoint&#xff09; 簇&#xff08;ClusterID&#xff09; 通信数据帧抓包分析 接收过程中的端点和簇&#xff08;接收模块&#xff09; 接收过程中的端点 接收过程中的簇 发送过程中的端点和簇&#xff08;发送…

java适配器模式

一、是什么&#xff1f; 定义: 将一个类的接口变成另外一个类所期待的另一个接口, 从而使因接口不匹配而无法一起工作的两个类能够一起工作 举个例子, 苹果手机想用type-c的充电器充电, 但充电接口不吻合, 所以就选哦一个转接头, 使type-c 能给苹果手机充电, 这就是适配器 …

物联网应用中的 Wi-Fi 6

近年来&#xff0c;设备智联在我们的日常生活中越来越常见。从智能家居设备到工业自动化系统&#xff0c;物联网技术正在改变我们与世界交互的方式。随着物联网设备的不断增多&#xff0c;对可靠、高容量和低功耗无线连接的需求变得尤为迫切。这就是 Wi-Fi 6&#xff08;即 802…

SpringBoot整合Mybatis-plus项目完成CRUD

一、准备阶段&#x1f349; 1.创建项目&#x1f95d; 2.引入依赖&#x1f95d; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-inst…

2023上半年软考系统分析师科目一整理-24

2023上半年软考系统分析师科目一整理-24 IEEE 802.1x是一种&#xff08; &#xff09;认证协议。 A.用户ID B.报文 C. MAC地址 D. SSID IEEE802.1X协议实现基于端口(MAC地址(的访问控制。认证系统对连接到链路对端的请求者进行认证。一般在用户接入设备上实现802.1X认证。在认证…

【MySQL】利用SQL短路,解决无数据表连接问题

系列文章 MySQL安装教程&#xff08;详细&#xff09; 本文链接&#xff1a;https://blog.csdn.net/youcheng_ge/article/details/126037520 MySQL卸载教程&#xff08;详细&#xff09; 本文链接&#xff1a;https://blog.csdn.net/youcheng_ge/article/details/129279265 …

代理服务器之 squid、lvs、nginx、haproxy之间的区别

代理服务器之 squid、lvs、nginx、haproxy之间的区别 代理服务可简单的分为正向代理和反向代理 1、正向代理 正向代理服务器&#xff1a;squid 用于代理内部网络对 Internet 的连接请求(如 VPN/NAT)&#xff0c;客户端指定代理服务器,并将本来要直接发送给目标 Web 服务器的 HT…

Spring Boot 中的 ElasticsearchRepository 是什么,原理,如何使用

Spring Boot 中的 ElasticsearchRepository 是什么&#xff0c;原理&#xff0c;如何使用 简介 Elasticsearch 是一个开源的分布式搜索和分析引擎&#xff0c;可以通过 RESTful API 进行访问。Spring Data Elasticsearch 是 Spring Data 项目的一部分&#xff0c;提供了与 El…

Java线程等待唤醒的三种方法

线程等待唤醒的三种方法 需求&#xff1a;我们实现A线程等待B线程执行完在执行。 Object下面的wait()和notify() 使用Object中的wait()方法让线程等待&#xff0c;使用Object中的notify()方法唤醒线程 public static void main(String[] args) throws InterruptedException…

提高企业云服务性价比,亚马逊云科技为用户提供全面和深入的算力支持

6月27日至28日&#xff0c;2023亚马逊云科技中国峰会于上海顺利召开。在本次峰会上&#xff0c;似乎找寻到了云计算领域竞争对手均日渐成熟&#xff0c;而亚马逊云科技却能一直保持领先地位的原因——过去的十几年里&#xff0c;亚马逊云科技“基于客户需求&#xff0c;快速进行…

django-vue-admin ubuntu 20.04 环境准备 记录

django-vue-admin 运行记录 https://django-vue-admin.com/document/hjbs.html https://django-vue-admin.com/document/hjbs.html https://bbs.django-vue-admin.com/article/9.html https://gitee.com/liqianglog/django-vue-admin/tree/demo_project 1. 安装 ubuntu-20.04…

【八股】【C++】函数与类

这里写目录标题 形参与实参的区别函数调用过程指针和引用当函数参数引用作为函数参数有哪些好处回调函数友元函数重载匹配运算符重载直接初始化与拷贝初始化函数指针C中struct&#xff08;结构&#xff09;和class&#xff08;类&#xff09;的区别C有哪几种构造函数构造函数的…

中华太极图

python代码&#xff1a; import turtle turtle.circle(100) turtle.color(black,black) turtle.begin_fill() turtle.circle(50,180) turtle.circle(-50,180) turtle.right(180) turtle.circle(100,180) turtle.end_fill() turtle.penup() turtle.goto(0,25) turtle.pendown()…

06_pinctr子系统与gpio子系统

目录 pinctrl子系统简介 I.MX6ULL的pinctrl子系统驱动 PIN驱动程序讲解 设备树中添加pinctrl节点模板 gpio子系统简介 I.MX6ULL的gpio子系统驱动 GPIO驱动程序简介 gpio子系统API函数 设备树中添加gpio节点模板 与gpio相关的OF函数 LED实验 LED灯驱动程序编写 运行…