第七章第三节(快速排序)

快速排序:快速排序是已知在实践中最快的排序算法,它的平均运行时间是O(N logN)
快速排序也是一种分治的递归算法。
快速排序的步骤:对数组S排序
1.如果S中元素个数是0或1,则返回。
2.取S中任一元素u,称为枢纽元(pivot)
3.将S-{u}(S中的其它元素)分为两个不相交的集合:S1 = { x ∈ S-{u} | x<= u} 和 S2= { x ∈ S-{u} | x >= u };
4.返回 quickSort(S1),u,quickSort(S2)
选取枢纽元策略:对S[left],S[right],S[center],适当的排序,三元素中的最小值放在S[left],最大值放在S[right]
中间值放在S[center],选取S[center]为枢纽元。可以将主元放到A[right-1]的位置,在分割阶段将i,j初始化为left+1,right-2。
快速排序对于小数组的排序,效率不是很高,解决方法是对于小数组的排序不递归的使用快速排序
使用插入排序这样对于小数组有效的排序算法

void Swap(int *a,int *b) {		//交换两个变量的值
	int t;
	t = *a;
	*a = *b;
	*b = t;
}
//选取枢纽元函数
int Median3(int a[], int left, int right) // 选取主元
{
	int centre = (left + right) / 2;
	if (a[left] > a[centre])
	{
		Swap(&a[left], &a[centre]);
	}
	if (a[left] > a[right])
	{
		Swap(&a[left], &a[right]);
	}
	if (a[centre] > a[right])
	{
		Swap(&a[centre], &a[right]);
	}
	// 上述步骤,是将a[left],a[centre],a[right]排序,使a[left]<a[centre]<a[right];
	Swap(&a[centre], &a[right - 1]);  //把主元放在R-1的位置
	return a[right - 1]; // 返回主元
}
//快速排序,根据枢纽元,划分,递归调用
void QSort(int a[], int Left, int Right) {
	int i, j;
	int pivot;
	int Cutoff = 25;
	//数组个数小于25,则使用插入排序
	if (Cutoff<=Right-Left) {
		pivot = Median3(a, Left, Right);	//选择枢纽元
		i = Left;
		j = Right - 1;
		while (1) {		//根据枢纽元完成一次划分
			while (a[++i] < pivot);
			while (a[--j] > pivot);
			if (i < j) {
				Swap(&a[i], &a[j]);		//将左边大于枢纽元的和右边小于枢纽元的交换
			}
			else {
				break;
			}
		}
		Swap(&a[i], &a[Right - 1]);		//将枢纽元放到正确的位置
		QSort(a, Left, i - 1);			//分治解决左侧问题
		QSort(a, i + 1, Right);			//分治解决右侧问题
	}
	else {
		insertionSort(a+left,right-left+1);    //使用插入排序
	}
}
void quickSort(int a[],int N) {
	QSort(a, 0, N - 1);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值