基本快速排序及其两种优化(C语言数据结构)

基本快速排序及其两种优化(C语言数据结构)

1.前言

​ 众所周知,快速排序作为交换排序的门面,号称排序界的“飞毛腿”,在一些正式场合,快排也是一些地方的官方排序结构。今天主要介绍的是基本版的快速排序以及对于基本版快排的两种性能优化。在一些面试场所下,由于时间关系可直接介绍基本版快排,对于优化可直接介绍思路即可。

注意:1.本次快排均采用的是hoare版本的代码。

​ 2.思想使用递归实现。


2.图解

在这里插入图片描述

3.基本思想

  1. 定义一个数组a存放待排的序列,定义一个指向最左边和最右边的指针leftright,还要定义一个临时指针pivot(通常指向最左边或最右边,这里默认左边)。
void QuickSort(int* a, int begin ,int end ) //形参写上头尾begin end 方便递归
	int left = begin, right = end; 
	int pivot = left;  //这里默认左边

在这里插入图片描述


2.right从右往左走找比pivot指针所指元素小的,left从左往右走找比pivot指针所指元素大的。各找到一个元素后进行值交换,此后各找到一个后往复值交换,直至相遇。相遇后再把pivot和相遇的位置值交换。

while (left < right)	//开始走,相遇即停
	{
		while (left<right && a[right] >= a[pivot])  //right先走,找小
		{										
			right--;
		}
		while (left < right && a[left] <= a[pivot])  // left先走 找大
		{
			++left;
		}
		Swap(&a[left],&a[right]); //交换左右指针的值
	}
	Swap(&a[left], &a[pivot]);//相遇后交换相遇位置和pivot
	pivot = left; //重置

**注意:**第二次循环中还是加入left<right是为了防止left和right越界

在这里插入图片描述

在这里插入图片描述


3.以上一二步结合为一趟 ,这一趟走完交换后就可确定pivot的最终正确位置。此后完成排序还需走N趟(N为序列个数),把pivot左区间和右区间均交给子问题递归完成。

	QuickSort(a,begin,pivot-1); //左区间
	QuickSort(a,pivot+1,end);//右区间

在这里插入图片描述


4.基本版代码汇总

void QuickSort(int* a, int begin ,int end )  
{
	if (begin >= end)
		return;
	int left = begin, right = end;
	int pivot = left;
	while (left < right)
	{
		while (left<right && a[right] >= a[pivot])  //right先走,找小
		{										//防止第二次循环right越界
			right--;
		}
		while (left < right && a[left] <= a[pivot])  // left先走 找大
		{
			++left;
		}
		Swap(&a[left],&a[right]);
	}

	Swap(&a[left], &a[pivot]);
	pivot = left; //重置

	QuickSort(a,begin,pivot-1);
	QuickSort(a,pivot+1,end);
}

5.优化

​ 以上快排在解决随机的序列性能出众,可是当序列变为接近有序或者序列数很少时性能却显得有些乏力。下面提供两种方案来分别解决,相当于给基本版的加上一些**“模组”**,而使其性能在面对一些特殊情况时也不落下风。

5.1三数取中(解决接近有序)

​ 单独写一个函数让初始的pivot所指的值不会是最大和最小的,而更趋近于中间值,这样在面对接近有序的序列时性能能够得到显著提升。

int  GetMidIndex(int* a, int begin, int end)
{
	int mid = (begin + end)  //假定一个中间值 以下代码皆是返回中间值
	if (a[mid] > a[begin])
	{
		if (a[mid] < a[end])
		{
			return mid;	
		}
		else if (a[begin] > a[end])
			return begin;
		else
			return end;
	}
	else                    // a[begin] > a[mid]
	{
		if (a[begin] < a[end])
			return begin;
		else if (a[end] > a[mid])
			return mid;
		else
			return end;
	}

}

这个单独的GetMidIndex函数写完后插入在函数开头即可。

int mid = GetMidIndex(a, begin, end);
	Swap(&a[begin], &a[mid]);

5.2小区间优化(解决小序列)

在数量较少时尽量不适用递归,转而使用直接插入排序,性能上可减少百分之五十以上的递归次数从而提高运行效率。

if ((end + begin + 1) < 10) //判断大概数字在10以下,就用直接插入排序 否则还是使用快排
	{
		InsertSort(a + begin, end - begin + 1);
	}
else
{
    QuickSort(a,begin,end);  
}
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Arthur___Cui

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值