数据结构之排序(上)

片头

嗨,小伙伴们,大家好!我们今天来学习数据结构之排序(上),今天我们先讲一讲3个排序,分别是直接插入排序、冒泡排序以及希尔排序。

1. 排序的概念及其应用

1.1 排序的概念

排序:什么是排序呢?排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。

稳定性:假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的相对次序保持不变,即在原序列中,r[i] = r[j],且r[i]在r[j]之前,而在排序后的序列中,r[i]仍在r[j]之前,则称这种排序算法是稳定的;否则称为不稳定的。

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

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


一、插入排序

2.1.1 基本思想:

直接插入排序是一种简单的插入排序法,其基本思想是:

把待排序的记录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的记录插入完为止,得到一个新的有序序列。(简便记忆:将待排序的元素一个个插入到一个已经排好序的有序序列中,直到整个数列都有序为止)

emmm,听上去有点迷迷糊糊,我们来举个例子~

直接插入排序,大家平时都玩过,我们玩扑克牌时,最好让我们的牌按照从小到大的顺序排列,排好序之后,我们就方便出牌。

摸了一张牌后,怎么保证手里的牌是有序的?手里的牌本身是有序的,再摸一张牌,摸完后,插入到相应位置,继续保持手里的牌有序。

插入排序的思想:已经有一个有序序列,再摸一张牌起来,然后把它插入到合适的位置,保持它们继续有序。

数组的本质:最开始把第一个数当作是有序的,把后一个数(第二个数)往前插入;前2个数有序,第3个数插入;前3个数有序,第4个插入,依次类推......

往前怎么插入呢?如果插入的元素比前一个元素小,就将前一个元素往后挪动;如果插入的元素比前一个大,就放到前一个元素的后面。

当 前n-1个数有序,(第n个元素)最后一个元素插入,数组排序完成。

2.1.2 直接插入排序

当插入第 i (i>=1)个元素时,前面的array[0],array[1],....,array[i-1]已经排好序,此时用array[i]的排序码与array[i-1],array[i-2],...的排序吗顺序进行比较,找到插入位置即将array[i]插入,原来位置上的元素顺序后移。

动图演示:

那我们要怎么实现直接插入排序呢?

 排序这个部分,按2个步骤去完成:

①单趟 ②整体

单趟插入排序:

思想:把一个数往前面的有序区间插入,必须确保[0,end]区间是有序的。

比如,我在[0,end]这个区间是有序的,我要把end+1这个位置的值往[0,end]这个区间进行插入。

我们先假设arr数组里面已经存放了  "1" ,  "3" ,  "5" ,现在我们想要存放"7"

如果我们想要插入"2",该怎么做呢?

完整过程如下:

此时,"2"比"1"大,直接将"2"放到"1"的后面即可,换句话来说,将temp里面的值放入arr数组的end+1位置即可。

如果我们想要插入"0",该怎么做呢?

很简单,思路和刚才基本一致,我们一起来画一画图~

当执行到最后一次的时候,"0"比"1"小,因此将"1"往后挪动一位,同时end--,然后将"0"这个元素,也就是temp保存的值插入到 end+1 的位置,数组排序完成。

单趟插入排序的代码如下:

//直接插入排序
void InsertSort(int* a, int n) {
	//单趟
	int end;
	int temp = a[end + 1];        //将end+1位置的值保存到temp变量里面
	while (end >= 0) {    
		if (temp < a[end]) {      //如果temp的值比end位置的值小
			a[end + 1] = a[end];  //将end位置的值往后挪动
			end--;                //end继续找前一个元素
		}
		else {
			break;                //如果temp的值比end位置的值大,或者两者相等,跳出循环
		}
	}
	//跳出循环有2种情况:
	//①temp的值比end位置的值大或者两者相同,直接将temp的值放到end位置的后面(end+1位置)
	//②temp的值比所有的值都小,循环结束,此时end为-1
	//那我还是要将temp放到end后面, -1 + 1 = 0, 放到下标为0的位置

	a[end + 1] = temp;
}

单趟的插入排序我们已经知道了,那么整体的插入排序怎么写呢? 

我们可以发现,无论是上面2种情况的哪一种,插入的元素都是放到end的后面,也就是end+1的位置。基于这样一个原因,我们可以:

① 定义一个变量temp,把end+1位置的值保存起来

② 最坏的情况下,end<0即循环结束(end == -1,已经超出了数组的范围),就是插入的值比所有的数都小

③跳出循环有2种情况:

  • temp的值比end位置的值大或者两者相等,直接将temp的值放到end位置后面(end+1位置)
  • temp的值比所有的值都小,循环结束,此时end为-1。那我还是要将temp的值放到end位置的后面(end+1位置), -1 + 1 = 0,将temp的值放到下标为0的位置。

 综上,  ①end起始位置是0,默认[0,0]区间的元素是有序的。此时,end == 0,下标为0的元素当作是有序的,后一个元素向前插入;end == 1,前2个元素有序了,再把第3个元素往前插入;end == 2,前3个元素有序了,再把第4个元素往前插入,以此类推....... 当 end == n-2, 前n-1个数已经有序,第n个数往前插入,整个数组有序。

因此,我们可以定义一个变量j,  j 的范围在[0,n-2] , j的最后一个位置的值为 n-2 ,也就是说 j < n-1

2.1.3 直接插入排序的代码:
//直接插入排序
void InsertSort(int* a, int n) {
//end起始位置是0,初始时,[0,0]区间的元素是有序的
//end = 0,下标为0的元素当做是有序的,后一个元素往前插入
//end = 1,前2个元素有序了,再把第3个元素往前插入
//end = 2,前3个元素有序了,再把第4个元素往前插入
//.....
//end = n-2,当前n-1个元素已经有序了,将第n个元素往前插入,整个数组有序
	//j的范围[0,n-2], j == n-2 --> j < n-1
	for (int j = 0; j < n - 1; j++) {
		int end = j;
	//定义一个变量temp,把temp+1位置的值保存起来
		int temp = a[end + 1];	
	//最坏的情况: end < 0即循环结束(end == -1,已经超出了数组的范围)
	//插入的值比数组中所有的值都小
		while (end >= 0) {
	//如果temp的值比end位置的值小,将end位置的值向后挪动
			if (temp < a[end]) {
				a[end + 1] = a[end];
				end--;
			}
	//如果temp的值比end位置的值大或者两者相同,跳出循环
			else {
				break;
			}
		}
	//跳出循环有2种情况:
	//①temp的值比end位置的值大或者两者相同,直接将temp的值放到end位置的后面(end+1位置)
	//②temp的值比所有的值都小,循环结束,此时end为-1
	//那我还是要将temp放到end后面, -1 + 1 = 0, 放到下标为0的位置
	
		a[end + 1] = temp;
	}
}

二、冒泡排序

  2.2.1 基本思想

 冒泡排序是一种基本的排序算法,通过重复地比较相邻的两个元素,并且交换位置,将最大的元素逐渐"冒泡"到最后面。冒泡排序的思想是重复地遍历待排序的元素,每次遍历比较相邻的两个元素,如果他们的顺序错误就交换位置,直到没有需要交换的元素为止。

   具体实现时,首先从数组的第一个元素开始,与相邻的元素进行比较,如果顺序错误就交换位置,然后继续比较相邻的下一对元素,一直到数组的最后一个元素。这样一次遍历后,最大的元素就会"冒泡"到最后面。然后再从第一个元素开始,重复上述操作,直到整个数组都排好序。

动图展示:

单趟冒泡排序的思想: 把最大的数换到最后

如果i从1开始,i的最后一个位置为 n-1, 将前一个元素和当前元素进行比较,也就是将 a[i-1] 和 a[i] 进行比较,如果前一个元素比当前元素大,则交换。当 i == n,循环结束

如果i从0开始,i的最后一个位置为 n-2, 将当前元素和后一个元素进行比较,也就是将 a[i] 和 a[i+1] 进行比较, 如果当前元素比后一个元素大,则交换。当 i == n-1,循环结束

单趟冒泡排序的代码:

//2个数进行交换
void Swap(int* a, int* b) {
	int temp = *a;
	*a = *b;
	*b = temp;
}

//冒泡排序
void BubbleSort(int* a, int n) {
	//单趟
	//如果i从1开始,i的结束位置在n-1
	//将前一个元素a[i-1]和当前元素a[i]进行比较,如果前一个比当前元素大,进行交换
	//如果i从0开始,i的结束位置在n-2
	//将当前元素a[i]和后一个元素a[i+1]进行比较,如果当前元素比后一个元素大,进行交换
	for (int i = 1; i < n; i++) {
		if (a[i - 1] > a[i]) {
			Swap(&a[i - 1], &a[i]);
		}
	}
}

我们已经知道了单趟的冒泡排序是如何实现的,那么整体的冒泡排序怎么做呢? 

第一次冒泡,冒泡完毕后,结束位置在下标为 n-1 的位置(i < n);第二次冒泡,结束位置在下标为 n-2 的位置(i < n-1);第三次冒泡,结束位置在下标为 n-3 的位置(i < n-2);第四次冒泡,结束位置在下标为 n-4 的位置 (i < n-3)....... 当 i == 1 ( i < n - (n-2) )时,冒泡结束。

 2.2.2 冒泡排序的代码:
//2个数进行交换
void Swap(int* a, int* b) {
	int temp = *a;
	*a = *b;
	*b = temp;
}

//冒泡排序
void BubbleSort(int* a, int n) {
	//第一次冒泡,冒泡完毕后,结束位置在 n-1 --> i<n
	//第二次冒泡,结束位置在 n-2 --> i<n-1
	//第三次冒泡,结束位置在 n-3 --> i<n-2
	//第四次冒泡,结束位置在 n-4 --> i<n-3
	//....
	//当 i==1 时,冒泡结束。
	// i == 1 --> i == n-(n-1) --> i<2 --> i< n-(n-2)
	// j的范围:[0,n-2], j == n-2 --> j<n-1
	for (int j = 0; j < n - 1; j++) {
		for (int i = 1; i < n-j; i++) {
			if (a[i - 1] > a[i]) {
				Swap(&a[i - 1], &a[i]);
			}
		}
	}
}
算法优化:

如果遍历一遍数组后没有发生任何元素交换,说明每一个数,前一个都小于后一个,此时数组已经有序,排序就可以结束了。

优化过的代码如下:

//2个数进行交换
void Swap(int* a, int* b) {
	int temp = *a;
	*a = *b;
	*b = temp;
}

//升级版冒泡排序
void BubbleSort1(int* a, int n) {
	//第一次冒泡,冒泡完毕后,结束位置在 n-1 --> i<n
	//第二次冒泡,结束位置在 n-2 --> i<n-1
	//第三次冒泡,结束位置在 n-3 --> i<n-2
	//第四次冒泡,结束位置在 n-4 --> i<n-3
	//....
	//当 i==1 时,冒泡结束。
	// i == 1 --> i == n-(n-1) --> i<2 --> i< n-(n-2)
	// j的范围:[0,n-2], j == n-2 --> j<n-1
	for (int j = 0; j < n - 1; j++) {
	//定义变量flag,假设此时数组是有序的
		int flag = 1;	
		for (int i = 1; i < n - j; i++) {
			if (a[i - 1] > a[i]) {
				Swap(&a[i - 1], &a[i]);
	//如果发生了交换,说明数组此时是无序的,flag为0
				flag = 0;
			}
		}
	//如果没有发生交换,说明数组已经有序
	//不需要进行比较了直接跳出循环
		if (flag == 1)
			break;
	}
}

三、希尔排序(最小增量排序)

希尔排序又称为缩小增量排序。它也是插入排序的一种,由希尔于1959年提出。

2.3.1 基本思想

希尔排序的基本思想是将待排序的元素分成几个子序列进行排序,通过逐步缩小子序列的间隔,最终使整个序列变为有序,具体步骤如下:

(1)首先确定一个增量gap,通常为数组的一半,然后将数组分成gap个子序列。

(2)分别对这些子序列进行插入排序,即对每个子序列进行直接插入排序,这样每个子序列都是部分有序的。

(3)再次选择一个较小的增量gap,重复步骤(2),直到gap为1。

(4)最后进行一次增量gap为1的插入排序,完成排序。

动图演示:

 希尔排序的思想:改革直接插入排序,有什么方法能让数组接近有序呢?

我直接再来一趟插入排序。把排序分成2个部分,第1个部分: 预排序。预排序的目标是:让数组接近有序;第2个部分:插入排序,目标是:让整个数组有序。

什么是预排序呢?

预排序是指:分组插入排序。目标:大的数更快换到后面的位置,小的数更快换到前面的位置

 gap给多少的问题:

①gap越大,数据跳得越快,大的数更快换到后面位置,小的数更快换到前面位置,但是越不接近有序

②gap越小,数据跳得越慢,但是越接近有序,当gap == 1时,插入元素后就是有序。

2.3.2 希尔排序的代码:
//希尔排序
void ShellSort(int* a, int n) {
	int gap = 3;
	for(int j = 0; j < gap; j++) {
		for (int i = j; i < n - gap; i += gap) {
			int end = i;
			int temp = a[end + gap];
			while (end >= 0) {
				if (temp < a[end]) {
					a[end + gap] = a[end];
					end = end - gap;
				}
				else {
					break;
				}
			}
			a[end + gap] = temp;
		}
	}
}

算法优化:

//希尔排序
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 temp = a[end + gap];
			while (end >= 0) {
				if (temp < a[end]) {
					a[end + gap] = a[end];
					end = end - gap;
				}
				else {
					break;
				}
			}
			a[end + gap] = temp;
		}
	}
}
	

希尔排序特性总结:

  1. 希尔排序是对直接插入排序的优化。
  2. 当gap>1时都是预排序,目的是让数组更接近有序。当gap==1时,数组已经接近有序的了,这样就会很快。这样整体而言,可以达到优化的效果。我们实现后可以进行性能测试的对比。

希尔排序的时间复杂度不好计算,因为gap取值的方法很多,导致很难去计算,因此在很多书中给出的希尔排序的时间复杂度都不固定。


片尾

今天我们学习了3个排序,分别是直接插入排序,冒泡排序以及希尔排序,希望能对看完文章的友友们有所帮助!!!

点赞收藏加关注!!!

谢谢大家!!!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

阿里云 物联网平台 MQTT连接、数据传输

阿里云 物联网平台 MQTT连接、数据传输 1、设备连接阿里云 2、多设备之前的通信、数据流转 3、设备数据来源的读取。 基于C# winform 开发上位机&#xff0c;读取设备、仪器、MES或者电子元器件的数据&#xff0c;MQTT传输至阿里云平台&#xff0c;可视化界面构建界面&#…

JSpdf,前端下载大量表格数据pdf文件,不创建dom

数据量太大使用dom》canvas》image》pdf.addimage方法弊端是canvas超出 浏览器承受像素会图片损害&#xff0c;只能将其切割转成小块的canvas,每一次调用html2canvas等待时间都很长累积时间更长&#xff0c;虽然最终可以做到抽取最小dom节点转canvas拼接数据&#xff0c;但是死…

字节码基础

基本概念 java中的字节码&#xff0c;英文bytecode。是java代码编译后的中间代码格式。JVM需要读取并解析字节码才能执行相应的任务。java字节码是JVM的指令集。JVM加载字节码格式的class文件。校验之后通过JIT编译器转换成本机机器代码执行。 java字节码简介 1、java byteco…

指针的奥秘(四):回调函数+qsort使用+qsort模拟实现冒泡排序

指针 一.回调函数是什么&#xff1f;二.qsort函数使用1.qsort介绍2.qsort排序整型数据3.qsort排序结构体数据1.通过结构体中的整形成员排序2.通过结构体中的字符串成员排序 三.qsort模拟实现冒泡排序 一.回调函数是什么&#xff1f; 回调函数就是一个通过函数指针调用的函数。 …

华为机试打卡 HJ2 计算某字符出现次数

要机试了&#xff0c;华孝子求捞&#xff0c;功德 描述 写出一个程序&#xff0c;接受一个由字母、数字和空格组成的字符串&#xff0c;和一个字符&#xff0c;然后输出输入字符串中该字符的出现次数。&#xff08;不区分大小写字母&#xff09; 数据范围&#xff1a; 1≤&a…

47.乐理基础-音符的组合方式-连线

连线与延音线长得一模一样 它们的区别就是延音线的第三点&#xff0c;延音线必须连接相同的音 连线在百分之九十九的情况下&#xff0c;连接的是不同的音&#xff0c;如下图的对比&#xff0c;连线里的百分之1&#xff0c;以现在的知识无法理解&#xff0c;后续再写 在乐谱中遇…

Linux提权--定时任务--打包配合 SUID(本地)文件权限配置不当(WEB+本地)

免责声明:本文仅做技术交流与学习... 目录 定时任务 打包配合 SUID-本地 原理: 背景: 操作演示: 分析: 实战发现: 定时任务 文件权限配置不当-WEB&本地 操作演示: 定时任务 打包配合 SUID-本地 原理: 提权通过获取计划任务执行文件信息进行提权 . 1、相对路径和…

Linux常见指令2️⃣

目录 cp指令&#xff08;重要&#xff09; mv指令&#xff08;重要&#xff09; cat、tac head、tail指令&#xff08;重要&#xff09; 知识点 时间相关的指令 知识点&#xff1a; Cal指令 grep 指令 zip/unzip指令 知识点 cp指令&#xff08;重要&#xff09; 语法…

K-CU12和利时工控单元

K-CU12和利时工控单元。控制策略组态&#xff0c;使用专用的组态软件 人机界面HMI设计&#xff1a;操作员站画面设计&#xff0c;使用专用的组态软件 K-CU12和利时工控单元文件组态 2文档管理软件 在工程师站上进行系统组态的主要工作&#xff1a; K-CU12和利时工控单元。系统配…

1067: 有向图的邻接表存储强连通判断

解法&#xff1a; 定理&#xff1a;有向图G是强连通图的充分必要条件是G中存在一条经过所有节点的回路 跟上道题一样 这是错误代码 #include<iostream> #include<vector> using namespace std; int arr[100][100]; void dfs(vector<bool>& a,int u) {a…

数据分析的统计推断

数据分析的统计推断 前言一、提出问题二、统计归纳方法三、统计推断四、统计推断步骤如何进行统计推断统计推断的基本问题点估计区间估计总体方差已知总体方差未知 假设检验假设检验的假设显著性水平 五、检验统计量常见的检验统计量 六、检验方法七、拒绝域八、假设检验步骤九…

如何使用活字格批量导入照片到数据表

活字格是一款功能强大的电子表格软件&#xff0c;除了基本的表格计算功能之外&#xff0c;还提供了丰富的扩展功能&#xff0c;可以用来实现各种自动化操作。例如&#xff0c;我们可以使用活字格来批量导入照片到数据表中。 以下是具体的操作步骤&#xff1a; 在活字格工作表…

离线修复.dll,Microsoft Visual C++

在安装mysql时遇到下面的问题&#xff0c;如果是有网络的情况下微软管网下载安装就行了&#xff0c;用的服务器不允许连接互联网。 后面经过寻找&#xff0c;找到了一个修复工具&#xff0c;可一次修复所有的问题&#xff0c;特别好用分享给宝子们。 下载链接&#xff1a;http…

SM935,SM942,SM150和利时备件

SM935,SM942,SM150和利时备件。组态软件&#xff0c;可组态控制图、机柜布置图、电源分配图等&#xff0c;可编辑、编译、SM935,SM942,SM150和利时备件。工程师站组态的基本步骤&#xff1a;SM935,SM942,SM150和利时备件。 1. 根据生产现场的控制方案画出控制系统原理图 2. 根据…

Request请求数据 (** kwargs参数)

这里写目录标题 &#x1f31f;前言&#x1f349;request入门1. params2. data3. json4. headers5. cookies6. auth7. files8. timeout9. proxies10. allow_redirects11. stream12. verify13. cert &#x1f31f;总结 &#x1f31f;前言 在Python中&#xff0c;发送网络请求是一…

设计模式——结构型模式——代理模式(静态代理、动态代理:JDK、CGLIB)

目录 代理模式 代理模式简介 代理模式的分类 代理模式组成 代理模式的优缺点 静态代理 背景前置 编写代码 JDK动态代理 编写代码 使用Arthas分析JDK动态代理底层原理 CGLIB动态代理 编写代码 三种代理的对比 代理模式使用场景 代理模式 代理模式简介 代理模式属…

数据可视化第五天(读取文件获得男生女生身高信息,并且可视化在一个图像)

文件 需要学生文件的可以私信我 过程 利用numpy的loadtxt文件读取学号&#xff0c;性别&#xff0c;和身高。 import numpy as np import matplotlib.pyplot as pltfilename/Users/oommnn/Desktop/python学习/数据分析/网课资料/第04天/student-data.txtuser_infonp.dtype(…

智慧公厕:数据驱动的公共厕所智慧化管理

公共厕所作为城市基础设施的重要组成部分&#xff0c;对于城市居民的生活质量和城市形象有着不可忽视的影响。然而&#xff0c;传统的公共厕所管理模式存在诸多问题&#xff0c;如设施老化、卫生状况不佳等&#xff0c;严重限制了公众对于公共厕所的使用体验。随着大数据和智能…

EDA设计学习笔记2:STM32F103C8T6最小系统板的仿绘

今日开始仿制练习一个STM32F103C8T6最小系统板&#xff0c;通过对这个最小系统板的仿制&#xff0c;达到对自己PCB设计的练习的目的&#xff0c;最终目标是自己设计出一块PCB&#xff0c;做一个OLED的桌面小摆件...... 也不知道画出来能不能用..... 目录 主控芯片的搜索与放置…

proteus示波器不弹出来

运行后示波器没有弹出来 点击调试&#xff08;Debug&#xff09;在点击Digital Oscilloscope 完成