112.【C语言】数据结构之排序(详解插入排序)

目录

1.排序定义

2.插入排序

"插入"的含义

代码

函数框架

函数设计思路

以升序为例,分析插入的三种可能

单趟排序代码

优化后

将单趟排序代码嵌入到循环中

错误代码

两种改法

运行结果

时间复杂度


1.排序定义

使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作

2.插入排序

"插入"的含义

例如对于一个升序数列:3,4,5,11;现在要求插入7使得数列还能满足升序

显然改动后的数列应该是3,4,5,7,11

算法:设要插入的数字为k,从数列的最后一个元素开始,依次向前比较,直到满足

时,向原数列插入k

代码

小提示:写排序代码时,先写单趟排序,之后将其嵌入循环内,能较容易写出

函数框架

cpp 复制代码
void InsertSort(int* a, int n)
{

}

函数设计思路

设tmp为要插入的数字,end为有序数组的最后一个元素的下标,n为无序数组元素的个数,a为数组指针,将tmp插入到[0,end]的区间中,构成有序数列

以升序为例,分析插入的三种可能

①tmp大于所有已经排序的数

②tmp大小介于已经排序的数之间

③tmp小于所有已经排序的数

①tmp大于所有已经排序的数

例如有序数列int arr[]={1,3,5,8,10,21,37,40},此时arr[end]==40;现插入tmp==50,

第一次循环:tmp>arr[end],直接arr[end+1]=tmp,循环结束

②tmp大小介于已经排序的数之间

例如有序数列int arr[]={1,3,5,8,10,21,37,40},此时arr[end]==40;现插入tmp==22

第一次循环:arr[end]==40,tmp<a[end],于是arr[end+1]=arr[end],end--,准备下一次循环

第二次循环:arr[end]==37,tmp<arr[end],于是arr[end+1]=arr[end],end--,准备下一次循环

第三次循环:arr[end]==21,tmp>arr[end],满足条件,arr[end+1]=tmp,循环结束

③tmp小于所有已经排序的数

例如有序数列int arr[]={1,3,5,8,10,21,37,40},此时arr[end]==40;现插入tmp==0

省略前几次循环,第?循环,arr[end]==1,tmp<arr[end],于是arr[end+1]=arr[end],end--,准备下一次循环

如果进行了下次循环,end==-1,会造成越界访问,此时应该立即退出循环!执行arr[end+1]=tmp

显然退出循环有两种情况:1.end<0,即end==-1时 2.tmp>arr[end]

因此将所有可能的情况讨论全,循环退出的条件才能写全

单趟排序代码

cpp 复制代码
	int end;
	int tmp;
	while (end >= 0)
	{
		if (tmp > arr[end])
		{
			arr[end + 1] = arr[end];
			end--;

		}
		else
		{
			arr[end + 1] = tmp;
			break;
		}
	}
	if (end == -1)
	{
		arr[end + 1] = tmp;
	}

上面这样写略显麻烦,无论满足什么条件退出循环,最后都执行arr[end + 1] = tmp;

优化后

cpp 复制代码
	while (end >= 0)
	{
		if (tmp > arr[end])
		{
			arr[end + 1] = arr[end];
			end--;
		}
		else
		{
			break;
		}
	}
	arr[end + 1] = tmp;

将单趟排序代码嵌入到循环中

main.c写入

cpp 复制代码
int main()
{
	int arr[] = { 1,6,2,4,7,3,9,0,-1,5 };
	printf("排序前:");
	PrintSort(arr, sizeof(arr) / sizeof(arr[0]));
	InsertSort(arr,sizeof(arr)/sizeof(arr[0]));
	printf("排序后:");
	PrintSort(arr, sizeof(arr) / sizeof(arr[0]));
	return 0;

错误代码

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

问题出在越界访问,当i==n-1时,执行int tmp = arr[i + 1];就越界了(arr的最后一个元素是arr[n-1])!!

两种改法

改法1:改for的循环条件

cpp 复制代码
for (int i = 0; i < n-1; i++)

改法2:改tmp和end赋的初始值

cpp 复制代码
	for (int i = 0; i < n; i++)
	{
		int end = i-1;
		int tmp = arr[i];
    //省略......
    }

备注:循环的一开始,无序数组的首元素可构成一个有序数组,后面的数依次插入有序数组

运行结果

时间复杂度

最坏情况,每次内循环结束条件都为end==-1,时间复杂度

最好情况:提供的数组恰好是有序的,时间复杂度

结论:从时间复杂度来看,数组越接近有序,插入排序的时间复杂度越低,而希尔排序就是基于这个结论设计的,有关希尔排序的内容参见下篇文章

相关推荐
Byron Loong11 分钟前
Python+OpenCV系列:膨胀和腐蚀——图像形态学操作深度解析
开发语言·python·opencv
枫の准大一12 分钟前
【C++游记】string的使用和模拟实现
c语言·开发语言·c++·算法
lshzdq14 分钟前
【机器人】控制之稳定性判定: 李雅普诺夫Lyapunov (7) 判定是否是李函数,思维导图
线性代数·算法·机器学习
千千道15 分钟前
Qt 实现 UDP 广播的详细教程
开发语言·qt·网络协议·udp
java菜鸡加油22 分钟前
代码随想录-算法训练营-番外(图论02:岛屿数量,岛屿的最大面积)
java·算法·leetcode·深度优先·力扣·图论
Matlab精灵27 分钟前
MCW TDMA-MIMO雷达信号处理MATLAB仿真
开发语言·matlab·信号处理
java菜鸡加油28 分钟前
代码随想录-算法训练营-番外(图论01:图论理论基础,所有可到达的路径)
java·算法·leetcode·力扣·图论
心死翼未伤38 分钟前
python数据分析之爬虫基础:requests详解
开发语言·爬虫·python·http·数据挖掘·数据分析·pip
9毫米的幻想1 小时前
【Linux系统】—— 权限的概念
android·linux·服务器·c语言·c++·学习
時光心向阳1 小时前
【Hexo】Hexo基本使用
笔记