插入排序和希尔排序:用C语言打造高效的排序算法

news2024/10/7 13:15:42

在这里插入图片描述

插入排序

插入排序的思路就像是你在整理一堆扑克牌。你先拿起第一张牌,然后拿起第二张牌,把它插入到合适的位置,使得你手上的两张牌是有序的。接着,你再拿起第三张牌,也把它插入到合适的位置,使得你手上的三张牌是有序的。依此类推,直到你把所有的牌都拿到手上,这时你手上的牌就是有序的了。

如果我们要对一个数组a进行插入排序(假设按照升序排列),我们可以这样做:

首先,我们认为数组的第一个元素a[0]是有序的。然后,我们把第二个元素a[1]插入到合适的位置,使得a[0]和a[1]是有序的。接着,我们把第三个元素a[2]也插入到合适的位置,使得a[0]、a[1]和a[2]是有序的。以此类推,直到我们把最后一个元素a[n-1]也插入到合适的位置,这时整个数组就是有序的了。

具体来说,如果我们已经知道下标为[0, end]的元素是有序的,那么如何把下标为end+1的元素插入进去呢?我们可以先把a[end+1]保存在一个临时变量tmp中,因为它很快就会被覆盖掉。然后,我们从右向左依次比较tmp和a[end]、a[end-1]、a[end-2]……等等。如果tmp比某个元素小,就把那个元素向右移动一位,为tmp腾出空间。如果tmp不比某个元素小,或者已经到达数组的左边界了,就停止移动,并把tmp放在当前空出来的位置上。

最后我们只需要控制end的变化即可。一开始end为0,表示只有一个元素是有序的。然后end逐渐增加,每次增加1,直到end为n-2时停止。这时候[0, n-2]是有序的,只需要把最后一个元素a[n-1]插入进去就完成了排序。

void InsertSort(int* a, int n)
{
	for (int i = 0; i < n - 1; ++i)
	{
		// 单趟插入排序
		// [0, end]有序
		int end = i;
		// 插入end+1
		int tmp = a[end + 1];
		while (end >= 0)
		{
			if (tmp < a[end])
			{
				a[end + 1] = a[end];
				--end;
			}
			else
			{
				break;
			}
		}
		// 把tmp插入到此时的end后面
		a[end + 1] = tmp;
	}
}

希尔排序

希尔排序的思路就像是你在打扫一间房间。你不是一次性把所有的东西都放到正确的位置,而是先按照一定的间隔分成几个小组,每个小组内部进行一次插入排序,使得每个小组内部的东西都相对有序。然后你再缩小间隔,重新分组,再进行一次插入排序,使得每个小组内部的东西更加有序。你不断重复这个过程,直到间隔为1,也就是所有的东西都在同一个小组里,这时你再进行一次插入排序,就可以把所有的东西都放到正确的位置了。

如果我们要对一个数组a进行希尔排序(假设按照升序排列),我们可以这样做:

首先,我们选择一个合适的间隔gap,比如数组长度的一半。然后我们把数组分成gap个小组,每个小组由相隔gap个元素组成。例如,如果gap为3,那么第一个小组由a[0]、a[3]、a[6]……等等组成,第二个小组由a[1]、a[4]、a[7]……等等组成,第三个小组由a[2]、a[5]、a[8]……等等组成。接着我们对每个小组进行一次插入排序,使得每个小组内部的元素都相对有序。

然后我们缩小间隔gap,比如取原来的三分之一加一。这时我们又可以把数组分成新的gap个小组,每个小组由相隔gap个元素组成。例如,如果新的gap为1,那么第一个小组就是整个数组。然后我们再对每个小组进行一次插入排序,使得每个小组内部的元素更加有序。

最后我们重复这个过程,直到间隔gap为1时停止。这时我们只需要对整个数组进行一次插入排序,就可以把所有的元素都排好序了。

具体来说,我们可以用一个外层循环控制间隔gap的变化,每次取原来的三分之一加一,直到为1时停止。然后我们用一个中层循环控制每个小组内部的插入排序过程。在中层循环中,我们需要从0开始向右遍历数组,每次只移动一个元素,直到到达n-gap为止。这样就可以把数组分成gap个小组,每个小组由相隔gap个元素组成。最后我们用一个内层循环控制每个小组内部的元素插入过程。在内层循环中,我们需要先把当前元素a[end+gap]保存在一个临时变量tmp中,并把它和左边相隔gap个元素的元素a[end]比较。如果tmp比a[end]小,就把a[end]向右移动gap位,为tmp腾出空间。如果tmp不比a[end]小,或者已经到达数组的左边界了,就停止移动,并把tmp放在当前空出来的位置上。这样就完成了一次插入排序。

注意:gap在不断缩小的过程中,最后一定会变成1,这时候就相当于执行了一次普通的插入排序,这是为了确保数组在最后能够完全有序。

void ShellSort(int* a, int n)
{
	int gap = n; // 组距,即同组中两两间隔的距离
	while (gap > 1) // 保证gap已经是1时不会重复循环
	{
		gap = gap / 3 + 1; // 保证最后一次是1
		for (int i = 0; i < n - gap; ++i) // 防止end+gap越界,进行分组预排
		{
			// [0, end]同组的已经有序,插入end+gap
			int end = i;
			int tmp = a[end + gap];
			while (end >= 0)
			{
				if (tmp < a[end])
				{
					a[end + gap] = a[end];
					end -= gap;
				}
				else
				{
					break;
				}
			}
			// 插入到end后面,保持组距
			a[end + gap] = tmp;
		}
	}
}

总结

  1. 插入排序的思想:每次从未排序的数中选取一个,将其插入到已排序的数中合适的位置,保证插入后仍然有序。
  2. 希尔排序的思想:把数组按照一定的间隔gap分成若干组,每组的元素下标相差gap,然后对每组内的元素进行插入排序。随着gap的不断减小,每组内的元素越来越多,最终当gap减小到1时,整个数组就变成了一组,此时进行最后一次插入排序,就可以得到完全有序的数组。

感谢大家的阅读!

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

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

相关文章

安装pycharm安装激活教程,不想只使用一个月,也不想花钱?

前言 本文方法可以安装使用截止当前2023.1.3最新版本Pycharm&#xff0c;过程非常简单&#xff0c;按照下面的步骤来一分钟即可搞定。 1.下载安装 已经安装过的可以跳过该步骤&#xff01; 下载 到官网地址下载正版安装包JetBrains Pycharm官网下载地址 安装 开始安装 选择安…

vertical-align详细介绍

1.定义: vertical-align是一个CSS属性&#xff0c;用于指定行内元素或表格单元格中内容的垂直对齐方式。它可以应用于行内元素、表格单元格或表格单元格中的内容。 该属性可以接受多个值&#xff0c;包括关键字&#xff08;如top、middle、bottom&#xff09;&#xff0c;长度…

Autosar诊断实战系列11-UDS 0x27加解密服务配置开发介绍

本文框架 前言1. DcmDsd 0x27服务添加1.1 DcmDsdSubServices中添加子服务2. DcmDsp中DcmDspSecurity实战配置2.1 DcmDspSecurity实战配置2.2 尝试次数及DelayTime进一步理解前言 在本系列笔者将结合工作中对诊断实战部分的应用经验进一步介绍常用UDS服务的进一步探讨及开发中注…

Python3+requests搭建接口自动化测试框架

目录 一、接口自动化的意义&#xff08;为什么做这个框架&#xff09; 二、准备工作 三、框架流程及逻辑 四、各模块介绍 五、具体使用 一、接口自动化的意义&#xff08;为什么做这个框架&#xff09; 新版本上线时之前版本的功能需要进行回归测试&#xff0c;导致大量的…

对文件中的数据进行排序

#include<stdio.h> #include<stdlib.h> #include<time.h> int main()//生成1000个随机数据 {srand((size_t)time(NULL));FILE* fpfopen("d:/data.txt","w");if(!fp) return -1;for(int i0;i<1000;i)fprintf(fp,"%d\n",ran…

【Git】保姆级教程:如何在 GitHub 上传大文件(≥100M)?(含自己的操作流程)

文章目录 一、问题导读二、自己的实际操作流程2.1 准备工作2.2 初始化仓库2.3 安装git lfs&#xff08;一个仓库里面执行一次就好了&#xff09;2.4 跟踪一下你要上传&#xff08;push&#xff09;的文件或指定文件类型2.5 添加.gitattributes2.6 添加要上传&#xff08;push&a…

Jmeter上传文件接口测试

Jmeter上传文件接口测试 接口测试&#xff0c;想必大家都做过&#xff0c;但是上传文件的接口&#xff0c;可能就不知所措。其实呢&#xff0c;还是那么回事~ 一、接口的业务 在接口文档缺失的前提下&#xff0c;那就从抓包玩起~Fiddler或者F12都可以。 本次我们接口实现的…

springboot+mysql实现特产销售平台

本次设计任务是要设计一个藏区特产销售平台&#xff0c;通过这个系统能够满足藏区特产销售管理的管理功能。系统的主要包括首页、个人中心、用户管理、特产信息管理、特产分类管理、特产分类管理、特产评分管理、系统管理、订单管理等功能。 管理员可以根据系统给定的账号进行登…

Graphics笔记

1.简介 Graphics是System.Drawing里面的绘图类。 2.开始绘制 2.1.创建绘图对象&#xff0c;三种方法 在窗体或控件的Paint事件中&#xff0c;用PaintEventArgs创创建绘图对象(控件或窗体重绘时&#xff09; /// <summary>/// 1.在窗体或控件的Paint事件中&#xff0c…

以数字孪生技术服务社会,推动企业高质量发展

随着“十四五”规划和2035年远景目标纲要中提出&#xff0c;要加快数字化发展&#xff0c;建设数字中国。为建设数字孪生智慧城市提供了国家战略指引。 新兴技术是数字孪生发展的助推剂&#xff0c;臻图信息通过数字孪生技术及研发的实景三维渲染引擎&#xff0c;搭载自主研发的…

IP地址【图解TCP/IP(笔记十)】

文章目录 IP地址的基础知识IP地址的定义IP地址由网络和主机两部分标识组成IP地址的分类广播地址IP多播子网掩码 IPv4首部 IP地址的基础知识 在用TCP/IP通信时&#xff0c;用IP地址识别主机和路由器。为了保证正常通信&#xff0c;有必要为每个设备配置正确的IP地址。在互联网通…

使用Gradio库进行交互式数据可视化:Timeseries模块介绍

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

计算机体系结构基础知识介绍之硬件预测(二)

一、ROB详述 指令执行的四个步骤是&#xff1a; 发射&#xff08;Issue&#xff09;&#xff1a;从指令队列中获取一条指令。如果有空闲的保留站和ROB条目&#xff0c;就发射指令&#xff1b;如果寄存器或ROB中有可用的操作数&#xff0c;就发送操作数到保留站。更新控制字段…

gigachad1靶机详解

gigachad_vh靶机详解 扫描到ip后对ip做一个全面扫描&#xff0c;发现有一个匿名服务器&#xff0c;是可以免密登陆的。 登陆上后发现就一个文件&#xff0c;get到我们电脑上。 file一下发现是一个zip文件&#xff0c;unzip解压一下&#xff0c;发现给了一个用户名chad&#xf…

ESP32设备驱动-直流电机与L298N电机驱动器

直流电机与L298N电机驱动器 文章目录 直流电机与L298N电机驱动器1、L298N介绍2、硬件准备3、软件准备4、驱动实现在本文中,我们将介绍如何使用ESP32通过L298N电机驱动器驱动直流电机。 1、L298N介绍 L298N 电机驱动器模块非常易于与微控制器一起使用,而且相对便宜。 它被广泛…

基于springboot的微信小程序宠物领养医院系统(源代码+数据库+10000字论文)085

基于springboot的微信小程序宠物领养医院系统(源代码数据库10000字论文)085 一、系统介绍 本项目有网页版和小程序端 本系统分为管理员、医生、用户三种角色 用户角色包含以下功能&#xff1a; 登录、注册、宠物领养、医生在线咨询、查看挂号、个人中心、密码修改、宠物寄…

HTTP第18讲——HTTP的缓存控制

诞生背景 由于链路漫长&#xff0c;网络时延不可控&#xff0c;浏览器使用 HTTP 获取资源的成本较高。所以&#xff0c;非常有必要把“来之不易”的数据缓存起来&#xff0c;下次再请求的时候尽可能地复用。这样&#xff0c;就可以避免多次请求 - 应答的通信成本&#xff0c;节…

测试工程师简历编写指南,送简历模板

目录 概述 简历元素 1. 基础信息 2. 个人情况 3. 教育背景 4.专业技能 5.工作经历&项目经 6. 自我评价&兴趣爱好 7. 其他内容 总结&#xff1a; 概述 在人才市场中&#xff0c;一次完整的求职过程通常包括以下阶段&#xff1a; 简历筛选电话面试笔试面谈意向…

Linux信号机制-3

转自&#xff1a;深入理解Linux内核——signals | linkthinking 信号很早就在 unix 系统中出现了&#xff0c;它用于用户进程之间的交互&#xff0c;几十年以来&#xff0c;变化都不大。信号是一个发送给进程或者进程组的消息&#xff0c;它只是一个数字&#xff0c;没有参数或…

spring AOP中pointcut表达式详解

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是「奇点」&#xff0c;江湖人称 singularity。刚工作几年&#xff0c;想和大家一同进步&#x1f91d;&#x1f91d; 一位上进心十足的【Java ToB端大厂…