【数据结构基础】之八大排序(C语言实现)

news2025/1/15 23:48:56

【数据结构基础】之八大排序(C语言实现)

  • 🐧 冒泡排序
    • ♈️ 冒泡排序原理及代码实现
    • ♈️ 稳定性分析
  • 🐧 选择排序
    • ♈️ 选择排序原理及代码实现
    • ♈️ 稳定性分析
  • 🐧 插入排序
    • ♈️ 插入排序的原理及代码实现
    • ♈️ 稳定性分析
  • 🐧 希尔排序
    • ♈️ 希尔排序的原理及其代码实现
    • ♈️ 希尔排序的时间复杂度分析
    • ♈️ 稳定性分析
  • 🐧 堆排序
    • ♈️ 稳定性分析
  • 🐧 归并排序
    • ♈️ 归并排序递归版本
    • ♈️ 归并排序非递归版本
    • ♈️ 归并排序的时间复杂度和空间复杂度分析
    • ♈️ 稳定性分析
  • 🐧 快速排序
    • ♈️ 快速排序递归实现(Hoare版本-----多坑版)
    • ♈️ 快速排序递归实现(挖坑法)
    • ♈️ 快速排序递归实现(前后指针法)
    • ♈️ 快速排序非递归实现
    • ♈️ 快速排序的三个优化
      • 🎀 三数取中
      • 🎀 小区间优化
      • 🎀 三路划分加随机取数
    • ♈️ 快速排序时空复杂度分析
    • ♈️ 稳定性分析
  • 🐧 计数排序
    • ♈️ 原理及代码
    • ♈️ 时空复杂度
    • ♈️ 稳定性分析
  • 🐧 内排序与外排序
  • 🐧 性能测试与排序正确性测试
    • ♈️ 正确性测试
      • 🎀 冒泡排序
      • 🎀 选择排序
      • 🎀 插入排序
      • 🎀 希尔排序
      • 🎀 堆排序
      • 🎀 快速排序
      • 🎀 归并排序
      • 🎀 计数排序
    • ♈️ 性能测试

前言:算法和数据结构是有一定的关联的,八大排序算法在数据结构里算是比较重要的一个部分,今天本篇博客将介绍八大排序算法的原理和实现。
可视化工具及动画演示----旧金山大学(usfca)数据结构可视化工具

📃博客主页: 小镇敲码人
💞热门专栏:数据结构与算法
🚀 欢迎关注:👍点赞 👂🏽留言 😍收藏
🌏 任尔江湖满血骨,我自踏雪寻梅香。 万千浮云遮碧月,独傲天下百坚强。 男儿应有龙腾志,盖世一意转洪荒。 莫使此生无痕度,终归人间一捧黄。🍎🍎🍎
❤️ 什么?你问我答案,少年你看,下一个十年又来了 💞 💞 💞

🐧 冒泡排序

♈️ 冒泡排序原理及代码实现

冒泡排序应该我们大家绝大部分人都接触过,因为很简单所以常常出现在教科书中用来启发学生,它重复地访问要排序的元素序列,依次比较两个相邻的 元素,如果顺序(如从大到小、首字母从Z到A)错误就把他们交换过来。访问元素的工作是重复地进行,直到没有相邻元素需要交换,也就是说该元素序列已经排序完成。
这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”到序列的顶端(升序或降序排列),就如同碳酸饮料中二氧化碳的气泡最终会上浮到顶端一样,故名“冒泡排序”。----部分节选自百度百科

冒泡排序可视化演示及工具
代码实现:

// 冒泡排序
void BubbleSort(int* a, int n)
{
	int flag = 0;//设置标志变量优化
	for (int i = 0; i < n - 1; i++)//排序n个数,需要n-1趟
	{
		flag = 0;
		for (int j = 0; j < n-i-1; j++)
		{
			if (a[j] > a[j + 1])//如果满足交换条件交换
			{
				flag = 1;//如果存在交换,将flag设置为1
				Swap(&a[j], &a[j + 1]);
			}
		}
		if (flag == 0)//如果这一趟没有数交换说明已经有序,结束排序
		{
			break;
		}
	}
}
  • 冒泡排序排n个数(升序),每一趟排完后最大的数(剩下的数里面)都到了正确的位置,所以不用管了。只需要n-1个是因为,后面的数都到了正确的位置,那剩下一个数也一定在正确的位置。
    时间复杂度:O(N^2)

♈️ 稳定性分析

冒泡排序是一种稳定的算法排序,首先解释下什么叫做稳定性,稳定性就是在排序中,相同的元素在排序后仍然保持之前的相对位置,就好比1 2 1 4 3,使用冒泡排序升序排列后,那两个1的相对位置会发生变化吗,答案是不会,冒泡排序是相邻的元素你大于我或者你小于我就会交换,等于是不会交换的,所以冒泡排序是一种稳定的排序。

🐧 选择排序

♈️ 选择排序原理及代码实现

选择排序的工作原理(升序)是每次从剩下的元素序列里面找到最小的元素,放到已经有序的序列后面,直到没有元素。选择排序可视化演示及工具

代码实现:

// 选择排序
void SelectSort(int* a, int n)
{
	int i = 0;
	int j = 0;
	int mini = 0;//创建变量,保存当前最小值的下标
	for (i = 0; i < n-1; i++)//n-1趟就排完了
	{
		mini = i;
		for (j = i; j < n; j++)
		{
			if (a[mini] > a[j])//找出目前最小的元素下标
			{
				mini = j;
			}
		}
		if(mini != i)//如果这个下标不是i位置,就交换
		Swap(&a[mini], &a[i]);
	}
}

选择排序的时间复杂度也是O(N^2)。

♈️ 稳定性分析

选择排序不是一个稳定的排序,会改变相同元素的相对位置,升序排序序列3 2 3 1 4,我们将1和位置0的3交换,改变相同元素3的相对位置,破坏了稳定性,不稳定。插入可视化演示及工具

代码实现:

// 插入排序
void InsertSort(int* a, int n)
{
	//0~end有序
	int end = 0;
	int i = 0;
	while (end < n - 1)
	{
		int tmp = a[end + 1];
		for (i = end; i >= 0; i--)
		{
			if (a[i] > tmp)
			{
				a[i + 1] = a[i];
			}
			else
			{
				break;
			}
		}
		a[i + 1] = tmp;
		end++;
	}
}

🐧 插入排序

♈️ 插入排序的原理及代码实现

插入排序,又称为直接插入排序,顾名思义,核心思想肯定是围绕着插入展开,它的原理是:前n-1个元素已经有序的情况下,我将第n个元素正确插入进去,就能保证前n个元素是有序的,就这样走n-1躺就可以将元素序列排成有序。

代码实现:

// 插入排序
void InsertSort(int* a, int n)
{
	//0~end有序
	int end = 0;
	int i = 0;
	while (end < n - 1)
	{
		int tmp = a[end + 1];//插入位置的元素
		for (i = end; i >= 0; i--)//调整其到正确位置,将大于tmp的都往整体后挪动一位,然后插入tmp
		{
			if (a[i] > tmp)
			{
				a[i + 1] = a[i];
			}
			else
			{
				break;
			}
		}
		a[i + 1] = tmp;
		end++;
	}
}

时间复杂度是O(N^2)。注意:当数组是一个有序序列的时候,插入一个数据的时间复杂度就是O(N)。

♈️ 稳定性分析

插入排序是稳定的排序算法,因为我们只有大于待插入的值才会将其往后挪动覆盖,小于等于就结束挪动了,没有改变相同元素的相对位置。

🐧 希尔排序

♈️ 希尔排序的原理及其代码实现

希尔排序是插入排序的一种,在性能上比直接插入排序更为优秀,又称“缩小增量排序”(Diminishing Increment Sort),它的原理是预处理+直接插入排序。希尔排序可视化演示及工具

下面我们来具体分析一下希尔排序的原理:

在这里插入图片描述
我们一步步来,下面我们实现(gap为2时的)多组排序。

	//gap为2时的预处理
	int gap = 2;
	for (int i = 0; i < gap; i++)//一共有gap组
	{
		int end = i;//将end赋值为这组的初始位置下标
		while (end < n - gap)//这是一组的直接插入排序,多组的在外面套一个循环就行
		{
			int tmp = a[end + gap];
			int j = 0;
			for (j = end; j >= 0; j -= gap)
			{
				if (a[j] > tmp)
				{
					a[j + gap] = a[j];
				}
				else
				{
					break;
				}
			}
			a[j + gap] = tmp;
			end += gap;
		}
	}

仔细比对,gap等于1时,这个就是我们的直接插入排序。很多教材上也会这样去写,这两种代码的实际效果是等价的。这个就相当于上一组的直接插入排序还没排完,就来排另外一个,只不过调换了下顺序。

	int gap = 2;
	for (int end = 0; end < n-gap; end++)//多组同时插入排序
	{
		int tmp = a[end + gap];
		int j = 0;
		for (j = end; j >= 0; j -= gap)
		{
			if (a[j] > tmp)
			{
				a[j + gap] = a[j];
			}
			else
			{
				break;
			}
		}
		a[j + gap] = tmp;
	}

这个实现了我们来探讨一下gap到底应该取多少?

在这里插入图片描述

最终的希尔排序代码:

// 希尔排序
void ShellSort(int* a, int n)
{
    int gap = n;//令gap等于n
	int i = 0;//i为控制每次插入排序的开始的位置的变量
	int j = 0;//j为每次直接插入排序的变量
	while (gap > 1)
	{
		gap = gap / 3 + 1;//gap不为一个固定的值,预处理多次,让我们的分组插入的效果更加好,降低后面直接插入的时间
		for (i = 0; i < n - gap; i++)//gap为某一个值时的分组插入,这里我们使用多组同时走插入排序
		{
			int end = i;
			int tmp = a[end + gap];
			for (j = end; j >= 0; j -= gap)
			{
				if (tmp < a[j])//小于就把大的值往后移
				{
					a[j + gap] = a[j];
				}
				else//找到了,break
				{
					break;
				}
			}
			a[j + gap] = tmp;//将tmp赋值给正确位置
		}
	}
}

♈️ 希尔排序的时间复杂度分析

根据大量的实验算出,希尔排序算法的时间复杂度是约为N^1.3,约差于N*logN,但是已经比直接插入排序好太多了,看来预处理虽然看起来复杂还是很有必要的。

下面我们来画图分析一下它的时间复杂度的计算:

在这里插入图片描述

♈️ 稳定性分析

希尔排序是不稳定的排序,直接插入排序是稳定的排序,但是希尔排序的预处理是分组直接插入排序,可能会改变相同元素的相对顺序,是不稳定的排序。

🐧 堆排序

堆排序我们在数据结构堆中已经具体的介绍过这里不再重复叙述。堆排序可视化演示及工具

代码实现:

// 堆排序
void AdjustDown(int* a, int n, int root)
{
	int child = root * 2 + 1;
	while (child < n)
	{
		if (child + 1 < n && a[child] < a[child+1])//找出最大的孩子
		{
			child = child + 1;
		}
		if (a[root] < a[child])//如果根节点的值比最大的孩子小就交换
		{
			Swap(&a[root], &a[child]);
			root = child;
			child = root * 2 + 1;
		}
		else//否则调整完成
		{
			break;
		}
	}
}
//堆排序,向下调整
void Heapsort(int* a, int n)
{
	assert(a);


	for (int i = (n-1-1)/2; i >= 0; i--)//我们向上调整,原地建一个大堆
	{
		AdjustDown(a,n,i);
	}
    //将大堆最大的和最后一个元素交换,并向下调整
	for (int end = n - 1; end >= 0; --end)
	{
		Swap(&a[0], &a[end]);//将堆顶元素放到堆最后面去
		AdjustDown(a,end,i);//此时的end就代表我们的元素个数
	}
}

堆排序的时间复杂度O(NlogN)。建堆的消耗是O(N),排序的过程是NlogN。

♈️ 稳定性分析

堆排序不是一个稳定的算法,我们使用下面的例子来解释一下。

在这里插入图片描述

🐧 归并排序

归并排序的原理在于两个字,归和并,也对应两个操作步骤,递归和合并,其中递归也包含着分治的思想 ,即把一个相同的问题,划分为很多一样的子问题来解决,下面我们来归并排序的归并排序可视化演示及工具

♈️ 归并排序递归版本

画图分析:
在这里插入图片描述
我们在写递归代码的时候,应该申请一个和原数组一样大小的tmp数组,用于辅助合并,因为两个子数组比较之后,需要把小的那个值放到一个地方,如果放到原先的数组里面就可能会覆盖我们的值,合并完之后在把left~right这段区间排好的值拷贝到原数组,好进行下一大组的比较。

代码实现:

void MergeSort1(int* a, int* tmp, int left,int right)
{
	if (left >= right)//如果左边界大于右边界就可以返回了,没有划分的余地了,但是要考虑一种特殊情况,就是只有一个数据的情况
		return;
	int mid = left + (right - left) / 2;
	MergeSort1(a, tmp, left, mid);//分割左子数组
	MergeSort1(a, tmp, mid + 1, right);//分割右子数组
	//左右两边的数组已经有序,我们进行合并操作
	int idx = left;
	int begin1 = left;
	int end1 = mid;
	int begin2 = mid + 1;
	int end2 = right;
	while (begin1 <= end1 && begin2 <= end2)
	{
		if (a[begin1] <= a[begin2])
		{
			tmp[idx++] = a[begin1++];
		}
		else
		{
			tmp[idx++] = a[begin2++];
		}
	}
	//处理还没有遍历完的数组,把他们加到数组的后面
	while (begin1 <= end1)
	{
		tmp[idx++] = a[begin1++];
	}
	while (begin2 <= end2)
	{
		tmp[idx++] = a[begin2++];
	}
	memcpy(a + left, tmp + left, sizeof(int) * (right - left + 1));//将已经排好序的,再拷贝到原数组
}
// 归并排序递归实现
void MergeSort(int* a, int n)
{
	int* tmp = (int*)malloc(sizeof(int)*n);//开一个辅助数组,用于辅助我们的合并操作
	if (tmp == NULL)//如果申请空间失败直接返回
	{
		perror("malloc failed");
		exit(-1);
	}
	MergeSort1(a,tmp,0,n-1);//递归,进行归并排序
	free(tmp);//释放空间
	tmp = NULL;//置空
}

♈️ 归并排序非递归版本

我们写了递归的版本,就应该思考一下如何写出它的非递归版本。

在这里插入图片描述

代码实现:

void MergeSort2(int* a, int* tmp, int n)
{
	int gap = 1;//一路开始是11合并
	while (gap < n)
	{
		for (int i = 0; i < n; i += gap * 2)//每次要合并的元素是gap*2,假设有这么多元素
		{
			//定义两个合并区间的左右边界
			int begin1 = i;
			int end1 = i + gap-1;
			int begin2 = i + gap;
			int end2 = i + 2 * gap - 1;
			if (end1 >= n || begin2 >= n)//如果已经没有两个区间合并,那么就无法合并了,跳出循环
			{
				break;
			}
			if (end2 >= n)//如果只是第二个区间的右边界超出范围,那么就仍然可以合并,将其右边界设置为数组的最后一个值的下标就行
			{
				end2 = n - 1;
			}
			int idx = i;//开始合并
			while (begin1 <= end1 && begin2 <= end2)
			{
				if (a[begin1] <= a[begin2])
				{
					tmp[idx++] = a[begin1++];
				}
				else
				{
					tmp[idx++] = a[begin2++];
				}
			}
			while (begin1 <= end1)//哪个还有剩余就把哪个加到tmp中对于这段区间的后面
			{
				tmp[idx++] = a[begin1++];
			}
			while (begin2 <= end2)
			{
				tmp[idx++] = a[begin2++];
			}
			memcpy(a + i, tmp + i, sizeof(int) * (end2 - i + 1));//将已经排好序的这段区间重新拷贝到原数组区间,准备下一次合并
		}
		gap *= 2;//开始下一轮合并
	}
}
// 归并排序非递归实现
void MergeSortNonR(int* a, int n)
{
	int* tmp = (int*)malloc(sizeof(int)*n);//创建辅助数组,存某一段区间排好序的数据
	if (tmp == NULL)//如果申请空间失败就退出程序
	{
		perror("malloc failed");
		exit(-1);
	}
	MergeSort2(a, tmp,n);//进入归并排序
	free(tmp);//释放空间
	tmp = NULL;//置空
}

♈️ 归并排序的时间复杂度和空间复杂度分析

归并排序不像希尔排序,存在前面的预处理对后面的分组排序是有优化的问题,时间复杂度非常好算,我们画图来分析:

♈️ 稳定性分析

先说结论,归并排序是一个稳定的排序。

在这里插入图片描述

🐧 快速排序

快速排序,和它的名字一样,它确实很快。它是冒泡排序算法的改进,通过多次比较和交换来实现。下面我们来具体介绍一下,它的原理和代码实现。快速排序可视化演示及工具

♈️ 快速排序递归实现(Hoare版本-----多坑版)

快速排序是Hoare这个人在1962年提出的一种二叉树结构的交换排序算法,我们先来学习一下这个经典的版本。

它的思想是这样的:

在这里插入图片描述

代码实现(递归):

int PartSort1(int* a, int left, int right)
{
	int keyi = left;//记录关键元素的位置,方便最后的交换
	while (left < right)//如果left < right就退出循环
	{
		while (left < right && a[right] >= a[keyi])//先找小(严格找小),也要加上left < right的条件,防止越界 
		{
			right--;
		}

		while (left < right && a[left] <= a[keyi])//再找大(严格找大),同样的越界条件要加上
		{
			left++;
		}
		Swap(&a[left], &a[right]);//找到了交换
	}
	Swap(&a[keyi], &a[left]);//最后交换keyi元素和最后一个小的元素的位置,这样单趟就排完了,keyi位置的元素到了正确位置,不用管它了
	return left;
}

void QuickSort(int* a, int left, int right)
{
	if (left >= right)//一个元素或者区间不存在的情况递归就结束
		return;
	int mid = PartSort1(a, left, right);//[left,mid-1] mid [mid+1,right]
	QuickSort(a, left, mid - 1);//继续递归左边,执行相同的单趟排序的思路
	QuickSort(a, mid + 1, right);//继续递归右边,执行相同的单趟排序的思路
}

为什么说Hoare版本的快速排序存在着很多坑呢,我们画图来分析一下:

在这里插入图片描述
递归的这个过程我们也浅浅的画个图来分析一下吧。

在这里插入图片描述

我们来解释一下这里为什么一定要右边先找比key小的值:

在这里插入图片描述

♈️ 快速排序递归实现(挖坑法)

前面我们介绍了Hoare版本的快速排序,但是坑比较多,后面有人对其在思路上做了优化更好理解了。

我们画图来介绍一下:

在这里插入图片描述

代码实现:

// 快速排序挖坑法
int PartSort2(int* a, int left, int right)
{
	int key = a[left];//保存left位置的值
	int hole = left;//保存坑位的下标
	while (left < right)//结束条件
	{
		while (left < right && a[right] >= key)//严格找小
		{
			right--;
		}
		a[hole] = a[right];//把小的值放到坑位
		hole = right;//更新坑位
		while (left < right && a[left] <= key)//严格找大
		{
			left++;
		}
		a[hole] = a[left];//把大的值放到坑位
		hole = left;//更新坑位
	}
	a[hole] = key;//把key值赋给相遇时的坑位
	return hole;//返回最后key值的下标
}
void QuickSort(int* a, int left, int right)
{
	if (left >= right)//一个元素或者区间不存在的情况递归就结束
		return;
	int mid = PartSort2(a, left, right);//[left,mid-1] mid [mid+1,right]
	QuickSort(a, left, mid - 1);//继续递归左边,执行相同的单趟排序的思路
	QuickSort(a, mid + 1, right);//继续递归右边,执行相同的单趟排序的思路
}

♈️ 快速排序递归实现(前后指针法)

前后指针法也是快速排序的一种实现思路,我们也来介绍一下。

画图分析:

在这里插入图片描述

代码实现:

// 快速排序前后指针法
int PartSort3(int* a, int left, int right)
{
	int keyi = left;//保存key值的下标
	int prev = left;//prev指向小于等于key的指针
	int cur = prev + 1;//cur找小,找到了交换,没找到后移
	while (cur <= right)//这是程序继续的条件
	{
		if (a[cur] < a[keyi] && ++prev != cur)//防止相同的值交换
		{
			Swap(&a[cur], &a[prev]);
		}
		++cur;//不管哪种情况,cur都要++
	}
	Swap(&a[prev], &a[keyi]);//最后别忘了把key值和pre交换
	return prev;//返回key的正确下标
}

♈️ 快速排序非递归实现

下面我们来看一下快速排序的非递归如何实现,递归有栈溢出的风险,所以一个优秀的程序员应该具备把递归转为非递归的能力。

老规矩画图分析:

在这里插入图片描述

代码实现:

// 快速排序 非递归实现     
void QuickSortNonR(int* a, int left, int right)
{
	Stack ps;//创建栈对象
	StackInit(&ps);//初始化栈对象
	StackPush(&ps, right);//入根节点的区间值,先入右,再入左,这样我们拿的时候就可以先拿到左
	StackPush(&ps, left);
	while (!StackEmpty(&ps))//如果栈为空,排序完成
	{
		int left1 = StackTop(&ps);//拿到栈顶区间的左边边界
		StackPop(&ps);//pop掉左边边界
		int right1 = StackTop(&ps);//拿到栈顶区间的左边边界
		StackPop(&ps);//pop掉右边边界
		int mid = PartSort3(a, left1, right1);//走一趟快速排序,哪个版本都可以,这里我们用的前后指针
		if (right1 > mid + 1)//先入右边区间,如果右边区间存在且长度不为1的话
		{
			StackPush(&ps,right1);
			StackPush(&ps, mid + 1);
		}

		if (left1 < mid - 1)//再入左边区间,如果左边区间存在且长度不为1的话
		{
			StackPush(&ps, mid-1);
			StackPush(&ps, left1);
		}
	}
	StackDestroy(&ps);//销毁栈
}

♈️ 快速排序的三个优化

🎀 三数取中

正常情况下,我们单趟排序排完最理想的情况是那个key值每一次都是中位数,这样左右两边就均衡了,递归的最深层数是logN,但是如果数组是一个有序数组呢?那每一次key值如果选第一个递归的层数不就是N层了,每一次调整要O(N),那么就是O(N^2)的时间复杂度,太慢了,我们使用三数取中来优化一下。

三数取中就是在left、right、mid三个位置取一个中位数,然后把它和left位置的值交换,让它来作为key值,这样就不害怕有序的情况了。

代码实现:

int get_mid(int* a, int left, int right)
{
	int mid = (left + right) / 2;//中间下标的数
	if (a[left] < a[mid])//如果left位置的值小于mid位置的值
	{
		if (a[mid] < a[right])//如果right位置的值大于mid位置的值,那么mid位置的值就是第2大的(中位数)
		{
			return mid;
		}
		else if (a[left] > a[right])//left位置的值大于right位置的值,mid是最大的
		{
			return left;
		}

		else
			return right;
	}

	else//a[left] >= a[mid]
	{
		if (a[mid] > a[right])//mid位置的值大于right位置的值
			return mid;
		else if (a[left] < a[right])//left不是最大的
			return left;
		else
			return right;
	}
}

加在单趟里面是这样的:

int PartSort1(int* a, int left, int right)
{
	int mid = get_mid(a, left, right);//三数取中找中位数
	Swap(&a[mid], &a[left]);//交换
	int keyi = left;//记录关键元素的位置,方便最后的交换
	while (left < right)//如果left < right就退出循环
	{
		while (left < right && a[right] >= a[keyi])//先找小(严格找小),也要加上left < right的条件,防止越界 
		{
			right--;
		}

		while (left < right && a[left] <= a[keyi])//再找大(严格找大),同样的越界条件要加上
		{
			left++;
		}
		Swap(&a[left], &a[right]);//找到了交换
	}
	Swap(&a[keyi], &a[left]);//最后交换keyi元素和最后一个小的元素的位置,这样单趟就排完了,keyi位置的元素到了正确位置,不用管它了
	return left;
}

🎀 小区间优化

在这里插入图片描述
这里我们使用插入排序,因为插入排序每一次排序执行的次数不是完整的等差数列,它比冒泡排序和选择排序都要优秀,希尔排序在数据量小的情况下和插入排序差不多,归并排序有空间消耗,堆排序需要建堆,综合考虑我们使用插入排序来进行小区间优化。

//快速排序小区间优化
void QuickSort(int* a, int left, int right)
{
	if (right - left + 1 > 10)//区间长度大于10,去走递归
	{
		int mid = PartSort1(a, left, right);//[left,mid-1] mid [mid+1,right]
		QuickSort(a, left, mid - 1);//继续递归左边,执行相同的单趟排序的思路
		QuickSort(a, mid + 1, right);//继续递归右边,执行相同的单趟排序的思路
	}
	else//区间长度小于10,去走插入排序
	{
		InsertSort(a + left, right - left + 1);
	}
}

这里为什么是a+left呢?因为每个大区间不断的去分割,在很多位置都会产生长度小于10的区间:

在这里插入图片描述
排序这个小区间的数组,需要传它初始值的地址。

🎀 三路划分加随机取数

但是即使我们做了上面两种优化,也仍旧不够,下面的OJ题使用我们的优化版本就过不了。
912.排序数组

在这里插入图片描述

这里我们几乎上面的所有优化都用到了,提交看看结果如何:

在这里插入图片描述

这种重复值的情况,确实让我们的快排很难受,因为时间复杂度会到O(N^2),下面我们介绍第三种优化:
三路划分

画图分析:

在这里插入图片描述

代码实现:

//三路划分
void QuickSort1(int* a, int left, int right)
{
	if (left >= right)
		return;
	int mid = get_mid(a, left, right);//三数取中找中位数
	Swap(&a[mid], &a[left]);//把那个中位数放到最左边
	int key = a[left];
	int begin = left;//保存左边界
	int end = right;//保存右边界
	int  cur = left + 1;//定义cur,用于遍历
	while (cur <= right)
	{
		if (a[cur] < key)//如果小于key
		{
			Swap(&a[cur], &a[left]);
			cur++;
			left++;
		}

		else if (a[cur] > key)//如果大于key
		{
			Swap(&a[cur], &a[right]);
			right--;
		}
		else
		{
			cur++;
		}
	}
	QuickSort1(a, begin, left - 1);//递归左边区间
	QuickSort1(a, right + 1, end);//递归右边区间
}

提交结果:

在这里插入图片描述
可以看到还是过不了,所有用例的都通过了,但是加起来时间仍然太长了,这个时候我们可以猜到,力扣对快速排序做了一些针对,我们对三数取中部分继续优化:

因为我们三数取中取mid、left、right三个位置数中的中位数,如果测试用例故意针对,每次让我们取到次小的数,那么我们递归的深度还是会很大,所以我们需要优化,mid不在取中间位置的数,而是取left~right区间的随机下标,这样它就无法针对我们了。

三数取中优化:

int get_mid(int* a, int left, int right)
{
	int mid = left+rand()%(right-left+1);//left~right随机下标的数
	if (a[left] < a[mid])//如果left位置的值小于mid位置的值
	{
		if (a[mid] < a[right])//如果right位置的值大于mid位置的值,那么mid位置的值就是第2大的(中位数)
		{
			return mid;
		}
		else if (a[left] > a[right])//left位置的值大于right位置的值,mid是最大的
		{
			return left;
		}

		else
			return right;
	}

	else//a[left] >= a[mid]
	{
		if (a[mid] > a[right])//mid位置的值大于right位置的值
			return mid;
		else if (a[left] < a[right])//left不是最大的
			return left;
		else
			return right;
	}
}

我们再加上随机数种子,这题就可以过了!

在这里插入图片描述

加上小区间优化会更快。

♈️ 快速排序时空复杂度分析

快速排序由于我们可以进行一系列的优化,像三路划分、三数取中(随机数)、小区间优化等来避免最坏的情况,让树的高度控制在ogN内,又因为每趟排序的时间复杂度是O(N),所以每层的也是O(N),最远是logN层,所以时间复杂度是N*logN。至于空间复杂度,递归是会消耗栈空间的,同样的经过优化可以到logN级别

♈️ 稳定性分析

快速排序不是不稳定的排序。因为它的性能很快,牺牲了稳定性,我们以Hoare版本来举个例子:

在这里插入图片描述

🐧 计数排序

计数排序是一个特殊的排序,它在有时候性能很优秀,我们有必要去了解一下它。计数排序可视化工具及演示

♈️ 原理及代码

我们还是画图来分析:

在这里插入图片描述

代码实现:

// 计数排序
void CountSort(int* a, int n)
{
	int max = a[0];
	int min = a[0];
	for (int i = 0; i < n; i++)//找最大最小值
	{
		if (max < a[i])
		{
			max = a[i];
		}
		if (min > a[i])
		{
			min = a[i];
		}
	}
	int range = max - min + 1;//求出元素的范围
	int* count = (int*)malloc(sizeof(int) * range);//动态申请空间
	if (count == NULL)//如果申请失败
	{
		perror("malloc failed");
		exit(-1);
	}
	memset(count, 0, sizeof(int) * range);//初始化count数组
	for (int i = 0; i < n; i++)//遍历数组,计数
	{
		count[a[i] - min]++;
	}
	int index = 0;
	for (int i = 0; i < range; i++)//映射覆盖元素
	{
		while (count[i]--)
		{
			a[index++] = i + min;
		}
	}
	free(count);//释放堆上的空间
	count = NULL;//指针置空
}

♈️ 时空复杂度

我们看一个算法的时空复杂度不能简单的看它套了几层循环,像刚刚的那个计数排序的时间复杂度就是O(N+range).其中range代表数组中元素的范围。

在这里插入图片描述

这里range不能忽略,因为不知道元素的范围和N比谁大,当range <= N时,这个算法还算可以,但是当range >> N,这个算法的性能就不行了。而且计数排序只能排整数。空间复杂度是O(range)。所以计数排序只适用于小数据范围的排序。是小数据范围,而不是小数据规模。

♈️ 稳定性分析

计数排序是稳定的排序,在排结构体时,如果是排的是其中一个整形数据是可以做到让其它结构体成员稳定的,具体大家可以看看这篇技术博客,写的不错,博主就不在这里分析了。

🐧 内排序与外排序

内排序是指在内存中排序,外排序是指的是可以在磁盘中排序。

在这里插入图片描述

🐧 性能测试与排序正确性测试

♈️ 正确性测试

借助这个OJ题来测试一下我们七大排序算法的正确性:

🎀 冒泡排序

在这里插入图片描述
这里我们的冒泡排序过了9组测试用例,说明其排序的功能没啥问题,就是太慢了。

🎀 选择排序

在这里插入图片描述
选择排序也是同样的效果。

🎀 插入排序

在这里插入图片描述

同为一个量级的排序算法,我们的插入排序居然过了11组数据!

🎀 希尔排序

在这里插入图片描述

我们的希尔排序直接狠狠拿下了!果然O(N*logN)级别的排序算法就是不一样(崇拜。

🎀 堆排序

在这里插入图片描述

堆排序也直接过了!

🎀 快速排序

在这里插入图片描述

这里我们快速排序被针对了,导致很慢!还是我们优化了很多地方才过的。

🎀 归并排序

在这里插入图片描述
归并排序也过了!

🎀 计数排序

在这里插入图片描述

可以看到我们计数排序的实力还是杠杠的,当然这是因为力扣的测试用例没有针对计数排序,如果将数据范围搞的很极端,计数排序的性能就更不上了。

♈️ 性能测试

我们使用下面的代码来测试一下八大排序的性能,测性能的时候我们要调到Release版本下,优化更好。

void TestOP()
{
	int N = 10000;
	printf("N: %d\n", N);
	int* a1 = (int*)malloc(sizeof(int) * N);
	if (a1 == NULL)
	{
		printf("a1 malloc failed\n");
		exit(-1);
	}
	int* a2 = (int*)malloc(sizeof(int) * N);
	if (a2 == NULL)
	{
		printf("a2 malloc failed\n");
		exit(-1);
	}
	int* a3 = (int*)malloc(sizeof(int) * N);
	if (a3 == NULL)
	{
		printf("a3 malloc failed\n");
		exit(-1);
	}
	int* a4 = (int*)malloc(sizeof(int) * N);
	if (a4 == NULL)
	{
		printf("a4 malloc failed\n");
		exit(-1);
	}
	int* a5 = (int*)malloc(sizeof(int) * N);
	if (a5 == NULL)
	{
		printf("a5 malloc failed\n");
		exit(-1);
	}
	int* a6 = (int*)malloc(sizeof(int) * N);
	if (a6 == NULL)
	{
		printf("a6 malloc failed\n");
		exit(-1);
	}
	int* a7 = (int*)malloc(sizeof(int) * N);
	if (a7 == NULL)
	{
		printf("a7 malloc failed\n");
		exit(-1);
	}

	int* a8 = (int*)malloc(sizeof(int) * N);
	if (a8 == NULL)
	{
		printf("a8 malloc failed\n");
		exit(-1);
	}
	for (int i = 0; i < N; ++i)
	{
		a1[i] = rand() % N;
		a2[i] = a1[i];
		a3[i] = a1[i];
		a4[i] = a1[i];
		a5[i] = a1[i];
		a6[i] = a1[i];
		a7[i] = a1[i];
		a8[i] = a1[i];
	}

	int begin1 = clock();
	BubbleSort(a1, N);
	int end1 = clock();

	int begin2 = clock();
	SelectSort(a2, N);
	int end2 = clock();

	int begin3 = clock();
	InsertSort(a3, N);
	int end3 = clock();

	int begin4 = clock();
	HeapSort(a4,N);
	int end4 = clock();

	int begin5 = clock();
	QuickSort1(a5,0,N-1);//三路划分版本
	int end5 = clock();

	int begin6 = clock();
	ShellSort(a6, N);
	int end6 = clock();

	int begin7 = clock();
	MergeSort(a7, N);
	int end7 = clock();

	int begin8 = clock();
	CountSort(a8, N);
	int end8 = clock();

	printf("BubbleSort:%dms\n", end1 - begin1);
	printf("SelectSort:%dms\n", end2 - begin2);
	printf("InsertSort:%dms\n", end3 - begin3);
	printf("HeapSort:%dms\n", end4 - begin4);
	printf("QuickSort1:%dms\n", end5 - begin5);
	printf("ShellSort:%dms\n", end6 - begin6);
	printf("MergeSort:%dms\n", end7 - begin7);
	printf("CountSort:%dms\n", end8 - begin8);
}
int main()
{
	srand(time(NULL));
	TestOP();
	return 0;
}

函数clock()是返回程序运行到当前指令的时间:

在这里插入图片描述

这是1w整形数据的运行结果:

在这里插入图片描述

这是10w整形数据的运行结果:

在这里插入图片描述

可以看到10W个O(N^2)的排序就有点hold不住了,特别是我们的冒泡排序,居然跑了14秒!

这是100w整形数据的运行结果:

O(N^2)的算法跑太慢了,我们就不让它们上场了。

在这里插入图片描述

这是1000w整形数据的运行结果:

在这里插入图片描述

这是1亿整形数据的运行结果:

在这里插入图片描述

可以看到1亿整型数据除了我们的快速排序和计数排序稳在10秒内,其它N*logN级别的排序算法都超过了10秒,HeapSort更是达到了惊人的41秒多,其它O(N^2)的算法都上不了桌,这里可能出现内存不够用的情况,把编译器调到64位就行。

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

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

相关文章

【数据结构】猛猛干7道链表OJ

前言知识点 链表的调试技巧 int main() {struct ListNode* n1(struct ListNode*)malloc(sizeof(struct ListNode));assert(n1);struct ListNode* n2(struct ListNode*)malloc(sizeof(struct ListNode));assert(n2);struct ListNode* n3(struct ListNode*)malloc(sizeof(struc…

GAMMA数据处理问题(七)

phase_sim_orb报这个错是什么原因呢&#xff0c;说是我的hgt文件和模拟的干涉图行数不匹配&#xff0c;之前geocode生成hgt的参数不是在mli.par文件中看吗&#xff0c;为什么会出现行数不匹配的情况啊&#xff0c;难道不是par文件中里面看&#xff1f;&#xff1f;&#xff1f;…

力扣hot100:153. 寻找旋转排序数组中的最小值(二分的理解)

由力扣hot100&#xff1a;33. 搜索旋转排序数组&#xff08;二分的理解&#xff09;-CSDN博客&#xff0c;我们知道二分实际上就是找到一个策略将区间“均分”。对于旋转数组问题&#xff0c;在任何位置分开两个区间&#xff0c;如果原区间不是顺序的&#xff0c;分开后必然有一…

[ROS 系列学习教程] rqt可视化工具箱 - Topic工具

ROS 系列学习教程(总目录) 本文目录 一、Message Publisher二、Message Type Browser三、Topic Monitor 一、Message Publisher Message Publisher 可以通过可视化界面发布topic。 启动方法&#xff1a; 在 rqt 窗口依次点击 Plugins -> Topics -> Message Publisher 启…

抖音平台热销的本腾和新讯随身WiFi,哪个更靠谱,更值得购买?

经常有粉丝朋友摆脱小编测评一下在某短视频平台上面非常火爆的两款随身WiFi&#xff0c;本腾随身WiFi和新讯随身WiFi到底哪个更好。今天&#xff0c;小编就为大家带来最真实的体验测评。 一、外观和产品 这方面新讯要比本腾做的更好&#xff0c;本腾的设备相对单一一些。新讯则…

关于Java发邮件提醒写周报实现(二)代码编写

背景 由于公司每周都要写周报&#xff0c;而日常工作很忙&#xff0c;所以很容易忘记这件事件&#xff0c;因此开发一个写周报提醒的机器人&#xff0c;进行特定时间提醒是时候写周报了。 有一个大前提&#xff0c;本技术实现&#xff0c;本着不开通任何收费服务的态度去考察使…

入门linux之Ubuntu学习

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言1、介绍Ubuntu2、虚拟机目录解析3、常用指令ls&#xff1a;罗列当前目录文件信息对ls -l 的结果解析1.第一个字符2.每三个字符&#xff08;第一个字符后&#x…

服务器软件express搭建web服务器

文章目录 1.express 是什么2.路由2.1&#xff08;参数一&#xff09;读取用户的请求&#xff08;request&#xff09;2.2&#xff08;参数二&#xff09;给用户响应&#xff08;response&#xff09;2.3&#xff08;参数三&#xff09;next()函数&#xff08;传递请求到下一个处…

卡尔曼滤波跟踪自由落体的高度程序,带MATLAB例程

背景 已知一物体做自由落体运动,对其高度进行20次测量,测量值如下表:(g=9.80m/s2). 设高度的测量误差分布为: N(0, 1),该物体的初始高度h0和速度v0分布也为高斯分布,且: 试求该物体高度和速度的随时间变化的最优估计(matlab Kalman filters)。N(0, 1),该物体的初始高…

【QT入门】 Qt实现自定义信号

往期回顾&#xff1a; 【QT入门】图片查看软件(优化)-CSDN博客 【QT入门】 lambda表达式(函数)详解-CSDN博客 【QT入门】 Qt槽函数五种常用写法介绍-CSDN博客 【QT入门】 Qt实现自定义信号 一、为什么需要自定义信号 比如说现在一个小需求&#xff0c;我们想要实现跨ui通信&a…

计算机实体安全

计算机实体安全定义&#xff1a; 对场地环境、设施、设备和载体、人员采取的安全对策和措施。 一、计算机可靠性与故障分析 1.1 计算机的可靠性 可靠性 (狭义) ■计算机在规定时间与条件下完成规定功能的 概率 ■规定条件&#xff1a;环境条件&#xff0c;使用条件&#xff0…

计数质数——算法思路

题目链接&#xff1a;204. 计数质数 - 力扣&#xff08;LeetCode&#xff09; 会超时的代码&#xff1a; 使用了枚举的方法 方法一&#xff1a; public static int countPrimes(int n) {int ans1;if (n<2)return 0;for (int i3;i<n;i){boolean flagtrue;for (int j2;j&…

docker安装WireGuard服务

启动 WireGuard 如下异常 则是linux内核需要升级 $ wg-quick down wg0 $ wg-quick up wg0 Error: WireGuard exited with the error: Cannot find device "wg0" This usually means that your hosts kernel does not support WireGuard!at /app/lib/WireGuard.js:65…

【计算机毕业设计】基于ssm038框架的网上招聘系统的设计与实现lw7

基于ssm038框架的网上招聘系统的设计与实现lw7&#xff1a; 本课题是基于ssm框架&#xff08;springMVC,spring,mybatis)的招聘系统&#xff0c;是标准的MVC模式&#xff0c;将系统分为表现层、controller层、service层、DAO层四层&#xff0c;使用spring MVC负责请求的转发和视…

【Linux】从零认识进程 — 中下篇

送给大家一句话&#xff1a; 人一切的痛苦&#xff0c;本质上都是对自己无能的愤怒。而自律&#xff0c;恰恰是解决人生痛苦的根本途径。—— 王小波 从零认识进程 1 进程优先级1.1 什么是优先级1.2 为什么要有优先级1.3 Linux优先级的特点 && 查看方式1.4 其他概念 2…

【C语言】编译和链接----从源代码到可执行程序的转换【图文详解】

欢迎来CILMY23的博客喔&#xff0c;本篇为【C语言】文件操作揭秘&#xff1a;C语言中文件的顺序读写、随机读写、判断文件结束和文件缓冲区详细解析【图文详解】&#xff0c;感谢观看&#xff0c;支持的可以给个一键三连&#xff0c;点赞关注收藏。 前言 欢迎来到本篇博客&…

写作兔怎么用 #微信#其他#知识分享

写作兔是一款非常实用的论文写作工具&#xff0c;不仅具有查重和降重的功能&#xff0c;而且操作简单方便&#xff0c;使用起来非常便捷。那么&#xff0c;接下来就让我们一起来了解一下“写作兔怎么用”。 首先&#xff0c;要使用写作兔&#xff0c;你只需要在浏览器中输入写作…

背包dp模板

01背包 for (int i1; i<n; i) //物品 {for (int j1; j<V; j) //容积 { // 装得下 分为 1.装 2.不装if (j>v[i]) dp[j] max(tmp[j],tmp[j-v[i]]v[i]);else dp[j] tmp[j]; // 装不下第i个}for (int j1; j<V; j) tmp[j] dp[j]; //滚动数组 } 滚动数组 for (int …

网络安全知识核心之TCP与UDP区别

TCP 面向连接&#xff08;如打电话要先拨号建立连接&#xff09;提供可靠的服务;UDP 是无连接的&#xff0c;即发送数据之前不需要建立连接&#xff0c;;UDP 尽最大努力交付&#xff0c;即不保证可靠交付。&#xff08;由于 UDP 无需建立连接&#xff0c;因此 UDP 不会引入建立…

【漏洞复现】科立讯通信指挥调度平台editemedia.php sql注入漏洞

漏洞描述 在20240318之前的福建科立讯通信指挥调度平台中发现了一个漏洞。该漏洞被归类为关键级别,影响文件/api/client/editemedia.php的未知部分。通过操纵参数number/enterprise_uuid可导致SQL注入。攻击可能会远程发起。 免责声明 技术文章仅供参考,任何个人和组织使…