排序(插入,希尔,选择,堆,冒泡,快速,归并,计数)

本文中的Swap()函数都是下面这段代码

// 交换
void Swap(int* p1, int* p2)
{
	int tmp = *p1;
	*p1 = *p2;
	*p2 = tmp;
}

文章目录

  • 常见排序:
  • 一.插入排序
    • 1.直接插入排序:
    • 2.希尔排序:
  • 二.选择排序
    • 1.选择排序:
    • 2.堆排序:
  • 三.交换排序
    • 1.冒泡排序:
    • 2.快速排序:
      • 注(重要):三数取中
      • (1)Hoare法:
      • (2)挖坑法:
      • (3)前后指针法:
      • 快速排序递归版:
      • 快速排序非递归版:
  • 四.归并排序
    • 1.归并排序
      • 归并排序递归版:
      • 归并排序非递归版
  • 五.计数排序


常见排序:

在这里插入图片描述

一.插入排序

1.直接插入排序:

在这里插入图片描述
直接插入排序的基本思想:通过取一个数a与前面的数比较,a小则将前面的数后移,a继续向前比较,直到找到比a更小的数,插入到该位置。

代码实现:

// 直接插入排序  时间:O(N^2)   空间:O(1)
void InsertSort(int* a, int n)
{
	for (int i = 0; i < n - 1; i++)
	{
		int end = i;
		int tmp = a[end + 1];
		// [0,end]区间为有序的,排序好的有序区间
		// end+1位置的值插入到有序区间中
		while (end >= 0)
		{
			if (a[end] > tmp)
			{
				a[end + 1] = a[end];
				end--;
			}
			else
			{
				break;
			}
		}
		a[end + 1] = tmp;
	}
}

2.希尔排序:

在这里插入图片描述

希尔排序的基本思想:先选定一个整数gap,将需要排序的内容分为gap组,所有的距离为gap的在同一组,并对每一组内的数进行排序。然后重复上述操作,直到gap减为1

我们来看一下分步图:
在这里插入图片描述

// 希尔排序  时间:O(N^1.3)(大致)   空间:O(1)
void ShellSort(int* a, int n)
{
	int gap = n;
	while (gap > 1)
	{
		// +1保证最后一个gap一定是1
		// gap > 1 时是预排序
		// gap == 1 时是插入排序
		gap = gap / 3 + 1;
		for (int i = 0; i < n - gap; i++)
		{
			int end = i;
			int tmp = a[end + gap];
			while (end >= 0)
			{
				if (tmp < a[end])
				{
					a[end + gap] = a[end];
					end -= gap;
				}
				else
				{
					break;
				}
			}
			a[end + gap] = tmp;
		}
	}
}

希尔排序就是对直接插入排序的优化当gap > 1时都是在对数组进行预排序,当gap = 1时,此时的数组已经接近有序了,且当gap为1时,我们代入到希尔排序的代码中,再与上面的插入排序比较一下,我们就能发现是一样的。

二.选择排序

1.选择排序:

在这里插入图片描述
选择排序的基本思想:每一次从待排序的数据中选出最小(或最大值),将其放在前面,直到全部待排序的数据元素排完。

为了提高排序效率,我们也可以同时选出最小值和最大值将最小值放在前面,最大值放在后面。

// 选择排序  时间:O(N^2)   空间:O(1)
void SelectSort(int* a, int n)
{
	int begin = 0, end = n - 1;  // 定义首位元素地址,同时找到最大和最小的,分别放到首和尾

	while (begin < end)
	{
		int mini = begin, maxi = begin;
		for (int i = begin + 1; i <= end; i++)  // 找到最大和最小值的下标
		{
			if (a[i] < a[mini])
			{
				mini = i;
			}
			if (a[i] > a[maxi])
			{
				maxi = i;
			}
		}

		Swap(&a[begin], &a[mini]);
		// 若首元素最大,将最小值交换到最前面后,begin会与maxi重叠
		// 需要更新maxi,防止begin与maxi重叠后,将把最小值交换到最后
		if (begin == maxi)
		{
			maxi = mini;
		}
		Swap(&a[end], &a[maxi]);

		begin++;
		end--;
	}
}

2.堆排序:

堆排序需要我们对堆这个数据结构有一定了解。

在这里插入图片描述
堆排序时利用数据结构树(堆)进行排序的一种算法。通过堆进行选择数据。排升序建大堆,排降序建小堆。

我们来看一下分步图:

在这里插入图片描述

// 堆排序  时间:O(NlogN)   空间:O(1)
void AdjustDown(int* a, int n, int parent)
{
	int child = parent * 2 + 1;

	while (child < n)
	{
		//找到两个孩子中,较小/较大 的孩子                                                                                                                                      
		if (a[child + 1] < a[child] && child + 1 < n)
		{
			child++;
		}
		if (a[child] < a[parent])
		{
			Swap(&a[child], &a[parent]);
			parent = child;
			child = parent * 2 + 1;
		}
		else
		{
			break;
		}
	}
}
void HeapSort(int* a, int n)
{
	// 向下调整建堆 O(N)
	for (int i = (n - 1 - 1) / 2; i >= 0; i--)
	{
		AdjustDown(a, n, i);
	}

	int end = n - 1;
	while (end > 0)
	{
		Swap(&a[0], &a[end]);
		AdjustDown(a, end, 0);
		end--;
	}
}

三.交换排序

1.冒泡排序:

在这里插入图片描述
冒泡排序的基本思路:从前向后,两两比较,小的放前,大的放后

// 冒泡排序  时间:O(N^2)   空间:O(1)
void BubbleSort(int* a, int n)
{
	for (int i = 0; i < n; i++)
	{
		int flag = 1;   // 标记设为1,若后面都有序则直接进行下一次循环
		for (int j = 0; j < n - i; j++)
		{
			if (a[j - 1] > a[j])
			{
				Swap(&a[j - 1], &a[j]);
				flag = 0;
			}
		}
		if (flag == 1)
		{
			break;
		}
	}
}

2.快速排序:

快排的代码我们采用分离处理,下面三种方法的代码只写明单趟排序。递归的部分在三种方法之后。

快速排序的基本思想:任取待排序元素中的某个元素作为基准值key,以这个基准值key将数组分割为两部分,左边的所有元素均小于基准值key,右边的所有元素均大于基准值key,然后将左边重复该过程,右边重复该过程,直到数组有序。

注(重要):三数取中

在选取基准值key的时候,如果数组本身是有序的,直接对左边或者右边取key,这时算法的时间复杂度就会退化O(N^2),因此递归的深度比较大,可能会导致栈溢出,这就很坑。所以,我们应该科学的选key,这种方法就是三数取中。

三数取中的基本思路:我们取最左边,中间,最右边三个值,对这三个值进行比较,选择中间大小的值作为key。

// 三数取中
int GetMidi(int* a, int left, int right)
{
	int midi = (left + right) / 2;
	// left midi right
	if (a[left] < a[midi])
	{
		if (a[midi] < a[right])
		{
			return midi;
		}
		else if (a[left] < a[right])
		{
			return right;
		}
		else
		{
			return left;
		}
	}
	else // a[left] > a[midi]
	{
		if (a[midi] > a[right])
		{
			return midi;
		}
		else if (a[left] < a[right])
		{
			return left;
		}
		else
		{
			return right;
		}
	}
}

(1)Hoare法:

在这里插入图片描述

在用Hoare法进行排序时,需要注意key的位置:左边做key,右边先走;右边做key,左边先走,这样才能保证相遇位置比key小。

在这里插入图片描述

我们来看一下分步图:
在这里插入图片描述

// 快速排序hoare版本
int PartSort1(int* a, int left, int right)
{
	int midi = GetMidi(a, left, right);
	Swap(&a[left], &a[midi]);  // 将key值放在左边

	int keyi = left;
	int begin = left, end = right;
	while (begin < end)
	{
		// 右边找小(右边先走)
		while (begin < end && a[keyi] <= a[end])
		{
			end--;
		}
		// 左边找大
		while (begin < end && a[keyi] >= a[begin])
		{
			begin++;
		}

		Swap(&a[begin], &a[end]);
	}

	Swap(&a[keyi], &a[begin]);
	return begin;
}

(2)挖坑法:

在这里插入图片描述

挖坑法的基本思路:先用key存储基准值,将第一个位置形成一个坑位,右侧找比key小的数,放入坑位,这个数的原位置形成新的坑位,重复这个操作,直到最后的坑位左侧都小于key,右侧都大于key。

注意:当此处为坑位时,不进行移动,移动另一个指针。

我们来看一下分步图:
在这里插入图片描述

// 快速排序挖坑法
int PartSort2(int* a, int left, int right)
{
	int midi = GetMidi(a, left, right);
	Swap(&a[left], &a[midi]);

	int key = a[left];
	int begin = left, end = right;
	int holi = left;
	while (begin < end)
	{
		while (begin < end && key <= a[end])
		{
			end--;
		}
		a[holi] = a[end];
		holi = end;
		while (begin < end && key >= a[begin])
		{
			begin++;
		}
		a[holi] = a[begin];
		holi = begin;
	}
	a[begin] = key;
	return begin;
}

(3)前后指针法:

在这里插入图片描述

前后指针法基本思路:cur找比key小的数,prev找比key大的数,然后进行交换,cur越界,最后将key和prev交换。

这种方法也不用和Hoare法一样考虑先走左还是先走右,也更易理解。

我们来看一下分步图:
在这里插入图片描述

// 快速排序前后指针法
int PartSort3(int* a, int left, int right)
{
	int midi = GetMidi(a, left, right);
	Swap(&a[left], &a[midi]);
	int keyi = left;

	int prev = left;
	int	cur = prev + 1;
	while (cur <= right)
	{
		if (a[cur] < a[keyi] && ++prev != cur)
		{
			Swap(&a[cur], &a[prev]);
		}
		cur++;
	}
	Swap(&a[keyi], &a[prev]);
	return prev;
}

快速排序递归版:

对上面三种方法进行递归,进行多次排序。

void QuickSort(int* a, int left, int right)
{
	if (left >= right)
		return;
	if ((right - left + 1) < 10)  // 这里小优化一下,数据内容小于10个的时候,直接进行插入排序,不走递归
	{
		InsertSort(a + left, right - left + 1);
	}
	else
	{
	    // 三种方式随意选
		int keyi = PartSort1(a, left, right);
		//int keyi = PartSort2(a, left, right);
		//int keyi = PartSort3(a, left, right);

		// [left, keyi-1] keyi [keyi+1, right]
		QuickSort(a, left, keyi - 1);
		QuickSort(a, keyi + 1, right);
	}
}

快速排序非递归版:

快排的非递归实现,需要用到栈这个数据结构,因此需要先了解栈。

快排的非递归基本思路:用栈模拟递归的实现,在栈中存储区域范围,然后取栈顶区间,单趟排序,右左子区间入栈,循环上面的操作,直到栈为空。

在这里插入图片描述

// 快速排序 非递归实现
#include"Stack.h"
void QuickSortNonR(int* a, int left, int right)
{
	Stack st;
	StackInit(&st);
	StackPush(&st, right);  // 先存right才能后用
	StackPush(&st, left);   // 后存left才能先用

	while (!StackEmpty(&st))  // 循环每走一次相当于之前的一次递归
	{
		int begin = StackTop(&st);
		StackPop(&st);
		int end = StackTop(&st);
		StackPop(&st);

		int keyi = PartSort1(a, begin, end);  // 三种方式随意选
		// [begin, keyi-1] keyi [keyi+1, end]
		if (keyi + 1 < end)
		{
			StackPush(&st, end);
			StackPush(&st, keyi + 1);
		}
		if (begin < keyi - 1)
		{
			StackPush(&st, keyi - 1);
			StackPush(&st, begin);
		}
	}

	StackDestroy(&st);
}

四.归并排序

1.归并排序

在这里插入图片描述

归并算法的基本思路:将已经有序的子序列合并,得到完全有序的序列;即先把每个子序列变成有序,然后两个子序列有序的合并成为一个新的有序子序列,最终合并为一个完整的有序序列。

我们来看一下分步图:
在这里插入图片描述

归并排序递归版:

// 归并排序    时间:O(NlogN)   空间:O(N)
//
// 归并排序递归实现
void _MergeSort(int* a, int* tmp, int begin, int end)
{
	if (begin >= end)
	{
		return;
	}

	int midi = (begin + end) / 2;  
	// [begin, midi] [midi + 1, end]  分为两个区间
    // 递归
	_MergeSort(a, tmp, begin, midi);
	_MergeSort(a, tmp, midi + 1, end);

	int begin1 = begin, end1 = midi;
	int begin2 = midi + 1, end2 = end;
	int i = begin;
	while (begin1 <= end1 && begin2 <= end2)
	{
		if (a[begin1] <= a[begin2])
		{
			tmp[i++] = a[begin1++];
		}
		else
		{
			tmp[i++] = a[begin2++];
		}
	}
	while (begin1 <= end1)
	{
		tmp[i++] = a[begin1++];
	}
	while (begin2 <= end2)
	{
		tmp[i++] = a[begin2++];
	}

	memcpy(a + begin, tmp + begin, (end - begin + 1) * sizeof(int));
}
void MergeSort(int* a, int n)
{
	int* tmp = (int*)malloc(sizeof(int) * n);
	if (tmp == NULL)
	{
		perror("malloc fail");
	}

	_MergeSort(a, tmp, 0, n - 1);

	free(tmp);
	tmp = NULL;
}

归并排序非递归版

归并排序的基本思路:首先设置gap表示归并的元素个数,然后对子序列中的gap个元素进行归并,更新gap,重复上面操作,以循环的形式模拟递归的过程。

我们来看一下分步图:
在这里插入图片描述

// 归并排序非递归实现
void MergeSortNonR(int* a, int n)
{
	int* tmp = (int*)malloc(sizeof(int) * n);
	if (tmp == NULL)
	{
		perror("malloc fail");
	}

	int gap = 1;  // 每组归并的数据个数
	while (gap < n)
	{
		for (int i = 0; i < n; i += 2 * gap)  // i表示每组归并的起始位置
		{
			// [begin1, end1][begin2, end2]
			int begin1 = i, end1 = i + gap - 1;
			int begin2 = i + gap, end2 = i + gap * 2 - 1;
			int j = i;
            
            // 这里需要对范围进行修正,否则有可能会导致数组越界访问
			// 第二组越界,整组都不需要归并
			if (begin2 >= n)
			{
				break;
			}
			// 第二组的尾end2越界,修正后,再归并
			if (end2 >= n)
			{
				end2 = n - 1;
			}

			while (begin1 <= end1 && begin2 <= end2)
			{
				if (a[begin1] <= a[begin2])
				{
					tmp[j++] = a[begin1++];
				}
				else
				{
					tmp[j++] = a[begin2++];
				}
			}
			while (begin1 <= end1)
			{
				tmp[j++] = a[begin1++];
			}
			while (begin2 <= end2)
			{
				tmp[j++] = a[begin2++];
			}

			memcpy(a + i, tmp + i, (end2 - i + 1) * sizeof(int));
		}

		gap *= 2;
	}

	free(tmp);
	tmp = NULL;
}

五.计数排序

在这里插入图片描述

计数排序的基本思想:计数排序又称为鸽巢原理,是对哈希直接定址法的变形应用。首先定义一个数组用来统计每个数出现的次数,然后根据统计的结果将序列回收到原来的序列中。

// 计数排序
void CountSort(int* a, int n)
{
	// 找最大,最小值,max - min + 1为count数组的长度,节省空间
	int min = a[0], max = a[0];
	for (int i = 1; i < n; i++)
	{
		if (a[i] < min)
		{
			min = a[i];
		}
		if (a[i] > max)
		{
			max = a[i];
		}
	}
	int range = max - min + 1;

	int* count = (int*)calloc(range, sizeof(int));
	if (count == NULL)
	{
		perror("malloc fail");
	}

	// 统计次数
	for(int i = 0; i < n; i++)
	{
		count[a[i] - min]++;
	}

	// 排序
	int j = 0;
	for (int i = 0; i < range; i++)
	{
		while (count[i]--)
		{
			a[j++] = i + min;  // 前面存储数据的时候减去了min,这里需要还原加上min
		}
	}

	free(count);
	count = NULL;
}

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

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

相关文章

docker部署rabbitMQ 单机版

获取rabbit镜像&#xff1a;我们选择带有“mangement”的版本&#xff08;包含web管理页面&#xff09;&#xff1b; docker pull rabbitmq:management 创建并运行容器&#xff1a; docker run -d --name rabbitmq -p 5677:5672 -p 15677:15672 rabbitmq:management --name:…

【OpenCV3】图像的翻转、图像的旋转、仿射变换之图像平移、仿射变换之获取变换矩阵、透视变换

1 图像的放大与缩小 2 图像的翻转 3 图像的旋转 4 仿射变换之图像平移 5 仿射变换之获取变换矩阵 6 透视变换 1 图像的放大与缩小 resize(src, dsize[, dst[, fx[, fy[, interpolation]]]]) src: 要缩放的图片dsize: 缩放之后的图片大小, 元组和列表表示均可.dst: 可选参数, 缩…

秋招春招,在线测评题库包含哪些?

各位小伙伴们&#xff0c;秋招春招的号角已经吹响&#xff0c;作为HR&#xff0c;我们又要开始忙碌起来了。面对众多的候选人&#xff0c;如何高效、准确地筛选出合适的人选呢&#xff1f; 在线测评就是一个非常有用的工具。本文就说说在线测评题库里的那些事儿&#xff0c;主…

ant-design-vue中实现a-tree树形控件父子关联选中过滤的算法

在使用ant-design-vue的框架时&#xff0c;a-tree是比较常用的组件&#xff0c;比较适合处理树形结构的数据。 但是在与后台数据进行授权交互时&#xff0c;就不友好了。 在原生官方文档的例子中&#xff0c;若子项被勾选&#xff0c;则父级节点会被关联勾选&#xff0c;但这勾…

天通报警呼叫柱:为边防哨所筑起坚固的通信堡垒

一、背景 边防哨所是国家安全的重要防线&#xff0c;肩负着守护边境安全、维护国家主权和领土完整的神圣使命。由于边防哨所通常位于地理位置偏远、环境恶劣的地区&#xff0c;通信问题成为影响边防工作的重要因素&#xff0c;给边防官兵的日常工作和应急响应带来了不小的挑战…

vue3封装数字上下滚动翻牌器,

优点&#xff1a;可以传入字符串设置初始数字位数&#xff0c;也可以直接传入数字&#xff0c;让他自己根据位数渲染 组件代码&#xff1a; <template><div class"count-flop" :key"compKey"><!-- --><div:class"item ! . ?…

欺诈文本分类检测(十四):GPTQ量化模型

1. 引言 量化的本质&#xff1a;通过将模型参数从高精度&#xff08;例如32位&#xff09;降低到低精度&#xff08;例如8位&#xff09;&#xff0c;来缩小模型体积。 本文将采用一种训练后量化方法GPTQ&#xff0c;对前文已经训练并合并过的模型文件进行量化&#xff0c;通…

判断奇偶数的小妙招

要判断一个数是奇数还是偶数&#xff0c;一般首先想到的都是对2取余&#xff0c;但其实有更高明的算法。 首先咱们要知道一个知识点&#xff1a;偶数的二进制末位为0&#xff0c;奇数的二进制末位为1。 这是进位制本身的规则决定的&#xff0c;二进制是“逢二进一”。如果末位…

Docker 学习 Day 2

docker 基本命令和操作 学习视频一、docker 常用命令1、帮助启动类命令2、镜像命令2.1、docker images2.2、docker search 某个 xxx 镜像的名字2.3、docker pull 某个 xxx 镜像的名字2.4、docker system df2.5、docker rmi 某个 xxx 镜像的名字 ID2.6、面试题&#xff1a;谈谈 …

谷歌seo网址如何快速被收录?

想让你的网站快速被搜索引擎收录&#xff0c;可以采取几种不同的策略。首先&#xff0c;确保你的网站内容丰富、有价值&#xff0c;搜索引擎更喜欢收录内容质量高的网站。同时&#xff0c;增强网站的外链建设&#xff0c;做好这些站内优化&#xff0c;接下来就是通过谷歌搜索控…

windows下自启springboot项目(jar+nginx)

1、将springboot项目打包为jar 2、新建文本文档 test.txt&#xff0c;并输入 java -jar D:\test\test.jar&#xff08;修改为自己的jar包位置&#xff09; 保存 然后修将后缀名改为 .bat 3、在同一目录再新建 文本文档test.txt&#xff0c;输入以下内容&#xff0c;&…

“杏鲍菇驱动机器人创新前行:康奈尔大学最新研究亮相Science子刊“

未来科技新篇章&#xff1a;杏鲍菇操控下的机器人奇旅&#xff01; 在这个日新月异的科技时代&#xff0c;你或许听说过机器人由AI驱动、由人脑操控&#xff0c;但你是否能想象&#xff0c;一颗看似平凡的杏鲍菇也能成为控制机器人的“大脑”&#xff1f; 没错&#xff0c;这不…

对抗性EM用于变分深度学习:在低剂量PET和低剂量CT中的半监督图像质量增强应用|文献速递--Transformer架构在医学影像分析中的应用

Title 题目 Adversarial EM for variational deep learning: Application to semi-supervised image quality enhancement in low-dose PET and low-dose CT 对抗性EM用于变分深度学习&#xff1a;在低剂量PET和低剂量CT中的半监督图像质量增强应用 01 文献速递介绍 医学影…

新专利:作物生长期预测方法及装置

近日,国家知识产权局正式授权了一项由北京市农林科学院智能装备技术研究中心、江苏省农业科学院联合申请的发明专利"作物生长期预测方法及装置"(专利号:ZL 2024 1 0185298.1)。该专利由 于景鑫 、任妮、吕志远、李友丽、吴茜等发明人耗时多年潜心研发&#xff0c;犹如…

6、关于Medical-Transformer

6、关于Medical-Transformer Axial-Attention原文链接&#xff1a;Axial-attention Medical-Transformer原文链接&#xff1a;Medical-Transformer Medical-Transformer实际上是Axial-Attention在医学领域的运行&#xff0c;只是在这基础上增加了门机制&#xff0c;实际上也就…

Java入门:08.Java中的static关键字01

1 static关键字 可以修饰属性变量&#xff0c;方法和代码段 static修饰的属性称为静态属性或类属性&#xff0c; 在类加载时就在方法区为属性开辟存储空间&#xff0c;无论创建多少个对象&#xff0c;静态属性在内存中只有一份。 可以使用 类名.静态属性 的方式引用 static修饰…

无人机动力系统设计之桨叶推力计算

无人机动力系统设计之桨叶推力计算 1. 源由2. 关键参数2.1 特性参数2.1.1 材质&#xff08;Material&#xff09;2.1.2 叶片数量&#xff08;Number of Blades&#xff09;2.1.3 重量&#xff08;Weight&#xff09;2.1.4 噪音水平&#xff08;Noise Level&#xff09; 2.2 安装…

一文为你详解期权波动率是什么?

今天期权懂带你了解一文为你详解期权波动率是什么&#xff1f;采用合适的期权组合来对冲或利用波动率变化带来的机会。不同策略适用于不同的市场条件和投资目标。 期权波动率 假如我们为地震灾害去买一份保险&#xff0c;你认为什么样地震的保险费会更贵呢&#xff0c;是深圳…

备忘录模式memento

学习笔记&#xff0c;原文链接 https://refactoringguru.cn/design-patterns/memento 允许生成对象状态的快照并在以后将其还原。备忘录不会影响它所处理的对象的内部结构&#xff0c; 也不会影响快照中保存的数据。

2024年湖北交安ABC公路安全员C证报考今时不同往日

2024年湖北交安ABC公路安全员C证报考今时不同往日 之前的交安ABC&#xff0c;你爱搭不理&#xff0c;现在的交安ABC&#xff0c;你高攀不起。真的应了这句话。所以建筑行业考证要趁早&#xff0c;别等要求严了你想报却缺东少西的。湖北交安ABC大变天了。 2024年之前报考湖北交…