详解快排的四种方式

快速排序是Hoare于1962年提出的⼀种⼆叉树结构的交换排序⽅法,其基本思想为:任取待排序元素 序列中的某元素作为基准值,按照该排序码将待排序集合分割成两⼦序列,左⼦序列中所有元素均⼩ 于基准值,右⼦序列中所有元素均⼤于基准值,然后最左右⼦序列重复该过程,直到所有元素都排列 在相应位置上为⽌。

1、lomuto前后指针法

复制代码
void Swap(int* x, int* y)
{
	int tmp = *x;
	*x = *y;
	*y = tmp;
}

int _QuickSort(int* arr, int left, int right)
{
	int key = left, slow = left, fast = left + 1;
	while (fast <= right)
	{
		if (arr[fast] < arr[key] && ++slow != fast)
		{
			Swap(&arr[fast], &arr[slow]);
		}
		fast++;
	}
	Swap(&arr[key], &arr[slow]);
	return slow;
}

void QuickSort(int* arr, int left, int right)
{
	if (left >= right)
	{
		return;
	}

	int key = _QuickSort(arr, left, right );
	//左子序列
	QuickSort(arr,left,key-1);
	//右子序列
	QuickSort(arr,key+1,right);
}

2、hoare版本

1)创建左右指针,确定基准值

2)从右向左找出⽐基准值⼩的数据,从左向右找出⽐基准值⼤的数据,左右指针数据交换,进⼊下次 循环

复制代码
int _QuickSort2(int* arr, int left, int right)
{
	int key = left;
	left++;
	while (left <= right)
	{
		//left找比key大的值
		while (left <= right && arr[left] < arr[key])
		{
			++left;
		}
		//left指向比key大的值

		//right找比key小的值
		while (left <= right && arr[right] > arr[key])
		{
			--right;
		}
		if (left <= right)
		{
			Swap(&arr[left++], &arr[right--]);
		}
	}
	Swap(&arr[key], &arr[right]);
	return right;
}

3、挖洞法

复制代码
int _QuickSort3(int* arr, int left, int right)
{
	int key = arr[left];
	int hole = left;
	while (left < right)
	{
		while (left < right && arr[right]>key)
		{
			right--;
		}
		arr[hole] = arr[right];
		hole = right;
		while (left < right && arr[left] < key)
		{
			left++;
		}
		arr[hole] = arr[left];
		hole = left;
	}
	arr[hole] = key;
	return hole;
}

4、非递归快排

复制代码
void QuickSortNoR(int* arr, int left, int right)
{
	ST st;
	STInit(&st);
	//[left,right],进栈时要right先进left后进这样出栈的顺序才是对的
	STpush(&st, right);
	STpush(&st, left);

	while (!StackEmpty(&st))
	{
		//[begin,end]
		int begin = STtop(&st);
		STpop(&st);

		int end = STtop(&st);
		STpop(&st);
		int key = begin;
		int slow = begin;
		int fast = begin + 1;
		while (fast<=end)
		{
			if (arr[fast] < arr[key] && ++slow != fast)
			{
				Swap(&arr[fast], &arr[slow]);
			}
			fast++;
		}
		Swap(&arr[slow], &arr[key]);
		key = slow;

		//左 [begin,key-1]
		//右 [key+1,end]

		if (key + 1 < end)
		{
			STpush(&st, end);
			STpush(&st, key + 1);
		}

		if (begin < key - 1)
		{
			STpush(&st, key-1);
			STpush(&st, begin);
		}

	}

}
相关推荐
晨曦夜月7 分钟前
map与unordered_map区别
算法·哈希算法
qeen8721 分钟前
【数据结构】建堆的时间复杂度讨论与TOP-K问题
c语言·数据结构·c++·学习·
图码31 分钟前
如何用多种方法判断字符串是否为回文?
开发语言·数据结构·c++·算法·阿里云·线性回归·数字雕刻
handler0140 分钟前
Linux 内核剖析:进程优先级、上下文切换与 O(1) 调度算法
linux·运维·c语言·开发语言·c++·笔记·算法
minglie11 小时前
实数列的常用递推模式
算法
我星期八休息1 小时前
IT疑难杂症诊疗室:AI时代工程师Superpowers进化论
linux·开发语言·数据结构·人工智能·python·散列表
代码小书生1 小时前
math,一个基础的 Python 库!
人工智能·python·算法
AI科技星1 小时前
全域数学·数术本源·高维代数卷(72分册)【乖乖数学】
人工智能·算法·数学建模·数据挖掘·量子计算
生成论实验室1 小时前
《事件关系阴阳博弈动力学:识势应势之道》第一篇:生成正在发生——从《即事经》到事件-关系网络
人工智能·科技·算法·架构·创业创新
漂流瓶jz1 小时前
UVA-1152 和为0的4个值 题解答案代码 算法竞赛入门经典第二版
数据结构·算法·二分查找·题解·aoapc·算法竞赛入门经典·uva