【排序算法】插入排序(希尔排序)

news2025/1/16 18:43:10

目录

一.直接插入排序

1.基本思想

 2.实现

3.特性

1.效率

2.时间复杂度:O(N^2)

3.空间复杂度:O(1)

4.稳定性:稳定

二.希尔排序

1.基本思想

2.实现

3.特性

1.效率

2.时间复杂度:O(N^1.3)

​编辑

3.空间复杂度:O(1)

4.稳定性:不稳定


一.直接插入排序

1.基本思想

直接插入排序是一种简单的插入排序法,其核心思想是对一个已经有序的序列插入一个数据,该数据依次比较有序序列中的值,直到插入到合适的位置。在我们玩扑克牌整理牌序的时候,用到的就是直接插入排序的思想。

 2.实现

直接插入排序的实现原理如下动图所示:

如上图所示,插入排序的前提是在一个已经有序的序列中进行插入,那么不妨假设在闭区间[0,end]中是有序的,那么插入元素在下标为end+1(用tmp存储)的位置开始插入,end+1和end的值进行比较,假设此时排升序,那么一旦end+1的值小于end,那么end的值就要向后移动,使end+1的值变为end就可(这里解释了为什么要用tmp存储end+1处的值,一旦被end替代,end+1的值就找不到了),当然,变换后end需要减1进行对下一个位置的比较,直到与0下标位置进行比较(此时end = 0)若不满足小于那么停止循环,直接在end+1的地方放置tmp即可。

值得注意的是,放置end+1为tmp必须要在循环外进行,这是由于边界问题,如下图所示插入2的情况:

void InsertSort(int* arr, int n)
{
	//[0,end]有序,end+1是插入值下标
	//最后一个插入的值下标为n-1,即end+1=n-1,end=n-2
	for (int i = 0; i < n - 1; i++)
	{
		int end = i;
		int tmp = arr[end + 1];
		while (end >= 0)
		{
			if (tmp < arr[end])
			{
				arr[end + 1] = arr[end];
				end--;
			}
			else
			{
				break;
			}
		}
		arr[end + 1] = tmp;
	}
}

3.特性

1.效率

元素越接近有序,直接插入排序算法的时间效率就越高,完全是逆序时效率最低

2.时间复杂度:O(N^2)

最好情况下序列为顺序,每个end+1只需要与前一个判断即可,此时时间复杂度为O(N)

最坏情况下序列为逆序,此时每个end+1都要换到下标为0为止,此时时间复杂度为O(N^2)

3.空间复杂度:O(1)

直接插入排序没有额外的空间开销,因此空间复杂度为O(1)

4.稳定性:稳定

二.希尔排序

1.基本思想

根据直接插入排序元素越接近有序,直接插入排序算法的时间效率就越高 的特性,希尔排序运营而生,希尔排序的核心思想就是优化直接插入排序,先对序列进行预排序,将逆序的状态破坏,达到一定的有序程度再进行直接插入排序,这就是希尔排序。

2.实现

定义gap(间隔)为一个具体的数,图中为3,那么整个序列将被分为gap组,分别对每组进行直接插入排序,就如图中黄红绿代表的三组一样,如此就能将一个“较为逆序”的序列变为“较为有序”,就像9这个最大的数一下就换到了最后一个位置,这样再进行插入排序,效率就会大大提高。

 其中对一组的插入排序写起来十分简单,只需要将直接插入排序的减1的操作改为减gap,加1改为gap即可,也就是说是上文插入排序的推广情况。然后再套上一个gap组的循环,就能实现预排序,注意要将第二层循环内改为i = j,不要只加个外层循环就完了。

void SheerSort(int* arr, int n)
{
	int gap = 3;

	//总共有gap组,每组都进行插入排序
	//注意i = j,每组的开始不同
	for (int j = 0; j < gap; j++)
	{
		//一组的插入排序
		//注意end+gap是插入值下标,i<n-gap
		for (int i = j; i < n - gap; i += gap)
		{
			int end = i;
			int tmp = arr[end + gap];
			while (end >= 0)
			{
				if (tmp < arr[end])
				{
					arr[end + gap] = arr[end];
					end -= gap;
				}
				else
				{
					break;
				}
			}
			arr[end + gap] = tmp;
		}
	}
	
	//最后进行直接插入排序
	InsertSort(arr, n);
}

当然也可以直接多组排序同时进行,这样就少套了一层循环,但执行的总次数和上述代码是相同的,也就是说效率依然是相同的。该循环直接从第一个数到下标为n-gap-1的值,该下标+gap即是最后一个值。 

void SheerSort(int* arr, int n)
{
	int gap = 3;

	//多组同时进行
	for (int i = 0; i < n - gap; i++)
	{
		int end = i;
		int tmp = arr[end + gap];
		while (end >= 0)
		{
			if (tmp < arr[end])
			{
				arr[end + gap] = arr[end];
				end -= gap;
			}
			else
			{
				break;
			}
		}
		arr[end + gap] = tmp;
	}


	//最后进行直接插入排序
	InsertSort(arr, n);
}

3.特性

1.效率

希尔排序是对直接插入排序的优化,当gap>1时就是预排序,目的是让数组更接近有序的状态,从而方便gap=1时的直接插入排序,提高效率。其中gap的取值方法有很多,但没有人证明哪种取值方法效率最高,以下出自《数据结构-用面相对象方法与C++描述》--- 殷人昆

2.时间复杂度:O(N^1.3)

由于gap取值不同,时间复杂度也不相同,但可以大概估算个平均结果是O(N^1.3)

以下内容出自《数据结构(C语言版)》--- 严蔚敏

3.空间复杂度:O(1)

希尔排序没有额外的空间开销,因此空间复杂度为O(1)

4.稳定性:不稳定

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

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

相关文章

AI在软件开发中的角色:辅助创新还是自动化取代?

文章目录 每日一句正能量前言&#xff1a;人工智能与软件开发的未来交汇点AI工具现状AI对开发者的影响工作方式的改变需要掌握的新技能保持竞争力的策略结论 AI开发的未来AI在软件开发领域的未来发展方向AI是否可能完全取代开发者如何在AI时代规划开发者的职业发展结论 后记&am…

【通过pnpm创建vite项目】

vue3最新项目技术构建后台管理系统 一、技术要求二、安装pnpm2.1 构建vite三、项目配置3.1 eslint 配置3.2 prettier配置3.3 stylelint配置3.4 配置husky3.5 配置commitlint3.6 pnpm 强制安装四、Element-plus 引入4.1 完整引入4.2 国际化配置4.3 配置别名4.4 Env环境配置4.5 s…

教育与社会的发展

生产力与教育的关系 政治经济制度与教育的关系 文化和人口与教育的关系

《梦醒蝶飞:释放Excel函数与公式的力量》11.4 ISERROR函数

第11章&#xff1a;信息函数 第四节 11.4 ISERROR函数 11.4.1 简介 ISERROR函数是Excel中的一个信息函数&#xff0c;用于检查指定单元格或表达式是否产生错误。如果单元格或表达式产生任何类型的错误&#xff08;如N/A、VALUE!、REF!等&#xff09;&#xff0c;则返回TRUE&…

子任务:IT运维的精细化管理之道

在当今的企业运营中&#xff0c;信息技术已成为支撑业务发展的核心力量。根据Gartner的报告&#xff0c;IT服务管理&#xff08;ITSM&#xff09;的有效实施可以显著提升企业的运营效率&#xff0c;降低成本高达15%&#xff0c;同时提高服务交付速度和质量。随着业务的复杂性和…

Python中对asyncio的实际使用

前言&#xff1a;一般涉及异步编程我都无脑用celery&#xff0c;但是最近在做一个项目&#xff0c;项目不大&#xff0c;也不涉及定时任务&#xff0c;所以就用了asyncio。 asyncio是python自带的模块&#xff0c;比celery轻量&#xff0c;使用起来也简单。以前学习过&#xf…

java中Error与Exception的区别

java中Error与Exception的区别 1、错误&#xff08;Error&#xff09;1.1 示例 2、 异常&#xff08;Exception&#xff09;2.1 示例 3、 区别总结 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; 当我们谈论编程中的错误&#xff08;Error&…

【LeetCode】917:翻转字符串

方法&#xff1a;双指针 class Solution { public:bool isletter(char ch){if(ch>a&&ch<z)return true;if(ch>A&&ch<Z)return true;return false;}string reverseOnlyLetters(string s) {int lens.size();int left0,rightlen-1;string s1;while(le…

60、基于浅层神经网络的数据拟合(matlab)

1、基于浅层神经网络的数据拟合的简介、原理以及matlab实现 1&#xff09;内容说明 基于浅层神经网络的数据拟合是一种常见的机器学习方法&#xff0c;用于通过输入数据来拟合一个非线性函数。这种方法通常包括一个输入层、一个或多个隐藏层和一个输出层。神经网络通过学习权…

【HarmonyOS】获取通讯录信息

【HarmonyOS】获取通讯录信息 一、问题背景&#xff1a; 在Android和IOS中&#xff0c;获取手机通讯录信息的方式&#xff0c;一般是申请通讯录权限后&#xff0c;获得手机所有的通讯录列表信息。 在鸿蒙中&#xff0c;因为权限方式安全性提高的变更&#xff1a;将用户权限限…

南京邮电大学运筹学课程实验报告1 图与网络求解 指导

一、题目描述 实验四 图与网络问题求解    实验属性&#xff1a; 设计型    实验目的 1&#xff0e;理解图的基本概念&#xff1b; 2&#xff0e;掌握运筹学软件的使用方法&#xff1b; 3. 掌握图中Dijkstra算法Matlab求解原理和方法。 …

系统概括javaScript运算符

目录 一.前言 二.算术运算符 三.前置后置递增运算符 四.比较运算符 五.逻辑运算符 六.各类运算符的优先级 一.前言 运算符通常被称为操作符&#xff0c;是用于实现赋值&#xff0c;比较和执行算术运算等功能的符号。 主要包括算术运算符&#xff0c;比较运算符&#xff0…

Vue 3 中创建一个动态的组件实例

本文将介绍如何在 Vue 3 中实现一个动态 Toast 组件实例。我们将创建一个简单的 Toast 组件&#xff0c;并使用一个动态创建实例的脚本来显示 Toast 消息。在 Vue 3 中创建动态组件实例有许多好处&#xff0c;这些好处主要体现在灵活性、性能、可维护性和用户体验等方面。 创建…

备考美国数学竞赛AMC8和AMC10:吃透1850道真题和知识点

距离接下来的AMC8、AMC10美国数学竞赛还有几个月的时间&#xff0c;实践证明&#xff0c;做真题&#xff0c;吃透真题和背后的知识点是备考AMC8、AMC10有效的方法之一。 通过做真题&#xff0c;可以帮助孩子找到真实竞赛的感觉&#xff0c;而且更加贴近比赛的内容&#xff0c;…

【python】PyQt5顶层窗口相关操作API原理剖析,企业级应用实战分享

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

2023年全国大学生电子信息竞赛E题——自动追踪系统(stm32和openmv+普通舵机)完美解决第四问

当时做的时候&#xff0c;当时看别人开源的23年的题&#xff0c;感觉一头雾水。两个字没思路。确实只有做了才会有思路。我这里清晰的整理出来思路。 1.第一问的复位问题就是写一个函数&#xff0c;如果按键按下&#xff0c;就进入&#xff0c;再按下就退出 当然这个复位是写死…

VMware Workstation 虚拟机网络配置为与主机使用同一网络

要将 VMware Workstation 虚拟机网络配置为与主机使用同一网络&#xff0c;我们需要将虚拟机的网络适配器设置为桥接模式。具体步骤如下&#xff1a; 配置 VMware Workstation 虚拟机网络为桥接模式 打开 VMware Workstation&#xff1a; 启动 VMware Workstation。 选择虚拟机…

某企业数据治理总体解决方案(45页PPT)

引言&#xff1a;集团企业数据治理总体解决方案旨在构建一个高效、安全、合规且灵活的数据管理体系&#xff0c;以支持企业决策优化、业务创新、风险管理和运营效率提升。该方案通过整合数据资源、规范数据流程、强化数据质量和促进数据共享&#xff0c;实现数据资产的最大化价…

【文档】软件详细设计说明书(直接套用word)

软件详细设计说明书直接套用实际项目。 原件可获取。

Golang | Leetcode Golang题解之第230题二叉搜索树中第K小的元素

题目&#xff1a; 题解&#xff1a; type MyBst struct {root *TreeNodenodeNum map[*TreeNode]int // 统计以每个结点为根结点的子树的结点数&#xff0c;并存储在哈希表中 }// 统计以 node 为根结点的子树的结点数 func (t *MyBst) countNodeNum(node *TreeNode) int {if…