草庐IT

快速排序详解

凉夏y 2023-11-05 原文

1.什么是快速排序

我们知道排序有很多种,常见的如希尔排序,插入排序,选择排序,堆排序等等,而快速排序也是排序家族中的一员。因为其在大多数情况下有着优秀的综合性能,快速排序的快速也算是实至名归,接下来就为大家讲解快速排序的思想与实现。

2.快速排序的核心思想

快速排序通过多次比较与交换来完成排序。而这个过程又被分为了多次重复单趟排序,接下来我们先从每一趟的排序讲起。

快速排序的单趟排序思想是:

在一个无序数组中取一个数key,每一趟排序的最终目的是:让key的左边的所有数小于key,key的右边都大于key(假设排升序)。

先不考虑这一步怎么实现,我们接着往下看。

以下面的数组为例,可以观察到的是,在完成单趟排序后,无论key的左边和右边是否有序,key都来到了它在整个数组有序时应该来到的位置,也就是这个数组的第四个位置。所以对于key来说,它已经被排好序了。

 接下来,我们对key的左右区间进行单趟排序,可以预见的是,每次排序都固定好了一个数。而当我们对细分的左右区间进行单趟排序,最终整个数组都会化为有序。

下面是快速排序的整体框架:

void QuickSort(int* a, int left, int right)
{
	if (left >= right)//如果区间只剩一个数或没有数就不进行操作
		return;
	int key = PartSort(a, left, right);//调用单趟排序函数,取key的位置
	QuickSort(a, left, key - 1);//递归调用,对左区间进行排序
	QuickSort(a, key + 1, right);//递归调用,对右区间进行排序
}

3.单趟排序三种方法

1.hoare方法

这个方法取自于快排的发明者本人

其单趟排序的思路是:取区间中最左或最右边的元素为key,定义两个变量,这里假设是p和q,q从区间的最右边向左走,找到比key小的元素就停下。p从最左边向右走,找到比key大的元素就停下。然后交换p和q所指向的元素。下面是p与q单次交换的示意图:

重复上面的过程,直到pq相遇,交换key和pq相遇位置的元素。

 

 这样,单趟排序就完成了。

可以看出,经过pq的不断交换,比key小的值换到了左边,比key大的值换到了右边。最终将key换至中间就完成了单趟排序。

这里的key可以取最左边的值,此时必须让最右边的q先走。因为在最后一个循环时,若是q向左走,撞上了p,p此时指向的元素是上一个循环结束后交换的值,这个值比key小。若是p向右走,撞上了q,那么在这个循环,q先走,并且q停下来了,所以q的位置是一个比key小的值。若让左边的p先走,则可能在最后交换key的步骤将一个大于key的值交换到最左边。具体原因不进行分析。

若药取最右边的值做key,则要让最左边的p先走,原因同上。 

 下面是代码实现:

int PartSort1(int* a, int left, int right)
{
	int key = left;//取最左边的元素做key
	while (left < right)//当左右没有相遇
	{
		while (left < right && a[right] >= a[key])//如果右比key小就退出循环
			right--;
		while (left < right && a[left] <= a[key])//如果左比key大就退出循环
			left++;
		swap(&a[left], &a[right]);//交换左右
	}
	swap(&a[key], &a[left]);//交换key和相遇位置的元素
	return left;//返回key的位置
}

2.挖坑法

这个方法单趟排序的思路是:取最左或最右边的元素为key,假设把这个位置“挖空”,让最右边的q向左走,直到遇到比key小的数,将其放到key的位置,自己的位置变“空”。直到pq相遇,那么这个位置就是最终的坑位,再将key填入这个坑位,就完成了一趟排序。

单趟交换示意图:

代码实现:

int PartSort2(int* a, int left, int right)
{
	int key = left;//取最左边的元素做key
	int x = a[left];//用变量x保存key的值
	while (left < right)
	{
		while (left < right && a[right] >= x)//从右往左找,找到比key小的值就停下来
			right--;
		a[key] = a[right];//将元素填入坑内
		key = right;//更新坑的位置
		while (left < right && a[left] <= x)//从左往右找,找到比key大的值就停下来
			left++;
		a[key] = a[left];//将元素填入坑内
		key = left;//更新坑的位置
	}
	a[key] = x;//最后将key的值填到坑内
	return key;//返回key的位置
}

3.快慢指针法

取最左边的数为key,定义两个快慢指针,都从key的下一位往右走,fast每走一步判断一下它指向的元素是否小于key,若小于则交换fast和slow位置的元素,并且让slow向前走,直到fast走到底,结束循环。最后让slow和key位置的值交换。再返回key的位置。

代码实现:

int PartSort3(int* a, int left, int right)
{
	int key = left;//取最左边的值为key
	int fast = key + 1;//定义快指针(实际是下标)
	int slow = key;//定义慢指针(实际是下标)
	while (fast <= right)
	{
		if (a[fast] < a[key])//当快指针指向元素小于key就交换
		{
			swap(&a[fast], &a[++slow]);//交换元素并且慢指针向前走
		}
		fast++;//快指针向前走
	}
	swap(&a[key], &a[slow]);//慢指针回退一位再交换
	return slow;//返回key的位置
}

注意这里的slow初始位置和fast不同,因为交换元素时要使用前置++,若使用后置++则要改写为下面的写法。因为如果每次交换都采用后置++,最终slow的位置会在期望位置的下一位,所以与key交换时要回退一位。而上面的更简洁,故采用上面的写法。

int PartSort3(int* a, int left, int right)
{
	int key = left;
	int fast = key + 1;
	int slow = key + 1;//起始位置不同
	while (fast <= right)
	{
		if (a[fast] < a[key])
		{
			swap(&a[fast], &a[slow++]);//采用后置++
		}
		fast++;
	}
	swap(&a[key], &a[--slow]);//需要回退一位
	return slow;
}

4.非递归实现快速排序

我们知道快速排序的思路就是将区间用key划分为左右两个小区间,再进行递归。

要用非递归的方式实现快速排序,我们可以借助栈来完成。

对于一个区间,我们可以化为栈中的两个元素进行表示:

比如表示[0,4]区间,我们可以先入0再入4。在取出时,我们先取到4,再取到0,这样就算是取出区间了(要注意出栈顺序与入栈顺序相反)。

具体的实现思路是:

先将整个区间入栈,然后进入循环体:取出区间,进行一趟快速排序,得到左右区间,再将左右区间入栈。直到整个栈为空就退出循环。

这样在每次循环中,就取出了要排序的区间,再将左右区间入栈,在之后的循环中排序,直到所有区间都被拆分排序到不可再拆分。

代码实现:

void QuickSortNonR(int* a, int left, int right)
{
	stack<int> st;//建立栈
	st.push(left);//将区间入栈
	st.push(right);
	while (!st.empty())//当栈不为空进入循环
	{
		right = st.top();//取区间最右值
		st.pop();//出栈
		left = st.top();//取区间最左值
		st.pop();//出栈
		int key = PartSort3(a, left, right);//对区间进行一趟排序,取得key值
		if (left < key - 1)//如果左区间可以再分,就入栈
		{
			st.push(left);
			st.push(key - 1);
		}
		if (key + 1 < right)//如果右区间可以再分,就入栈
		{
			st.push(key + 1);
			st.push(right);
		}
	}
}

5.两种优化快速排序的思想

1.三数取中

面对完全有序的数组,快速排序每趟排序后,key的位置都在边缘,每层递归都只能固定一个数,时间复杂度变成了O(N^2)。

面对这种情况,我们可以在取key时动手脚。每次取key我们不再只取最左或最右的值。而是对比最左、最右、中间的三个元素,取三个元素中,值在另外两者中间的元素作为key。这样,就打乱了有序数组,大大加快了快速排序在面对这种情况时的速度。

2.小区间优化

快速排序对一个元素不多的数组排序,仍需要进行多次递归调用,我们知道递归是比较消耗资源的,所以为了避免在快速排序递归的最后几层大量调用函数,我们可以在数组元素较少时不再递归,而是采用选择排序替代,这样就能在不损失多少速度的情况下减少大量的递归次数,达到优化速度的目的。

有关快速排序详解的更多相关文章

  1. ruby-on-rails - 需要帮助最大化多个相似对象中的 3 个因素并适当排序 - 2

    我需要用任何语言编写一个算法,根据3个因素对数组进行排序。我以度假村为例(如Hipmunk)。假设我想去度假。我想要最便宜的地方、最好的评论和最多的景点。但是,显然我找不到在所有3个中都排名第一的方法。Example(assumingthereare20importantattractions):ResortA:$150/night...98/100infavorablereviews...18of20attractionsResortB:$99/night...85/100infavorablereviews...12of20attractionsResortC:$120/night

  2. ruby-on-rails - 在具有 ActiveRecord 条件的相关模型中按字段排序 - 2

    我正在尝试按Rails相关模型中的字段进行排序。我研究的所有解决方案都没有解决如果相关模型被另一个参数过滤?元素模型classItem相关模型:classPriority我正在使用where子句检索项目:@items=Item.where('company_id=?andapproved=?',@company.id,true).all我需要按相关表格中的“位置”列进行排序。问题在于,在优先级模型中,一个项目可能会被多家公司列出。因此,这些职位取决于他们拥有的company_id。当我显示项目时,它是针对一个公司的,按公司内的职位排序。完成此任务的正确方法是什么?感谢您的帮助。PS-我

  3. ruby - 按数字(从大到大)然后按字母(字母顺序)对对象集合进行排序 - 2

    我正在构建一个小部件来显示奥运会的奖牌数。我有一个“国家”对象的集合,其中每个对象都有一个“名称”属性,以及奖牌计数的“金”、“银”、“铜”。列表应该排序:1.首先是奖牌总数2.如果奖牌相同,按类型分割(金>银>铜,即2金>1金+1银)3.如果奖牌和类型相同,则按字母顺序子排序我正在用ruby​​做这件事,但我想语言并不重要。我确实找到了一个解决方案,但如果感觉必须有更优雅的方法来实现它。这是我做的:使用加权奖牌总数创建一个虚拟属性。因此,如果他们有2个金牌和1个银牌,加权总数将为“3.020100”。1金1银1铜为“3.010101”由于我们希望将奖牌数排序为最高的,因此列表按降序排

  4. ruby-on-rails - 在不重新查询数据库的情况下重新排序 Rails 中的事件记录? - 2

    例如,假设我有一个名为Products的模型,并且在ProductsController中,我有以下代码用于product_listView以显示已排序的产品。@products=Product.order(params[:order_by])让我们想象一下,在product_listView中,用户可以使用下拉菜单按价格、评级、重量等进行排序。数据库中的产品不会经常更改。我很难理解的是,每次用户选择新的order_by过滤器时,rails是否必须查询,或者rails是否能够以某种方式缓存事件记录以在服务器端重新排序?有没有一种方法可以编写它,以便在用户排序时rails不会重新查询结果

  5. ruby - 如何以表格格式快速打印 Ruby 哈希值? - 2

    有没有办法快速将表格格式的ruby​​哈希打印到文件中?如:keyAkeyBkeyC...1232343451253474456...其中散列的值是不同大小的数组。还是使用双循环是唯一的方法?谢谢 最佳答案 试试我写的这个gem(在表中打印散列、ruby对象、ActiveRecord对象):http://github.com/arches/table_print 关于ruby-如何以表格格式快速打印Ruby哈希值?,我们在StackOverflow上找到一个类似的问题:

  6. ruby-on-rails - 如何对对象数组进行排序? - 2

    我有一个对象如下:[{:id=>2,:fname=>"Ron",:lname=>"XXXXX",:photo=>"XXX"},{:id=>3,:fname=>"Dain",:lname=>"XXXX",:photo=>"XXXXXXX"},{:id=>1,:fname=>"Bob",:lname=>"XXXXXX",:photo=>"XXXX"}]我想按fname排序,不区分大小写,所以它会导致编号:1,3,2我该如何排序?我正在尝试:@people.sort!{|x,y|y[:fname]x[:fname]}但这没有任何效果。 最佳答案

  7. ruby - 使用自定义排序首选项对数组进行排序? - 2

    有人可以告诉我如何根据自定义字符串对嵌套数组进行排序吗?比如有没有办法排序:[['Red','Blue'],['Green','Orange'],['Purple','Yellow']]“橙色”、“黄色”,然后是“蓝色”?最终结果如下所示:[['Green','Orange'],['Purple','Yellow'],['Red','Blue']]它不是按字母顺序排序的。我很想知道我是否可以定义要排序的值以实现上述目标。 最佳答案 sort_by对于这种排序总是非常方便:a=[['Red','Blue'],['Green','Ora

  8. Ruby 将对象插入现有的已排序对象数组 - 2

    我有以下现有的Dog对象数组,它们按age属性排序:classDogattr_accessor:agedefinitialize(age)@age=ageendenddogs=[Dog.new(1),Dog.new(4),Dog.new(10)]我现在想插入一条新的狗记录,并将它放在数组中的正确位置。假设我想插入这个对象:another_dog=Dog.new(8)我想把它插入到数组中,让它成为数组中的第三项。这是一个人为的示例,旨在演示我特别想如何将一个项目插入到现有的有序数组中。我意识到我可以创建一个全新的数组并重新对所有对象进行排序,但这不是我的目标。谢谢!

  9. ruby - 如何排序不是简单的哈希(哈希的哈希) - 2

    我有一个这样的哈希{55=>{:value=>61,:rating=>-147},89=>{:value=>72,:rating=>-175},78=>{:value=>64,:rating=>-155},84=>{:value=>90,:rating=>-220},95=>{:value=>39,:rating=>-92},46=>{:value=>97,:rating=>-237},52=>{:value=>73,:rating=>-177},64=>{:value=>69,:rating=>-167},86=>{:value=>68,:rating=>-165},53=>{:va

  10. ruby - 根据值然后键对ruby中的哈希进行排序 - 2

    如何在ruby​​中先根据值然后根据键对散列进行排序?例如h={4=>5,2=>5,7=>1}将排序为[[7,1],[2,5],[4,5]]我可以根据值进行排序h.sort{|x,y|x[1]y[1]}但我不知道如何根据值进行排序,然后在值相同时键入 最佳答案 h.sort_by{|k,v|[v,k]}这使用了Array的事实混入Comparable并定义逐元素。注意上面等价于h.sort_by{|el|el.reverse}相当于h.sort_by(&:reverse)这可能会或可能不会更具可读性。如果你知道Hashes一般都是先

随机推荐