草庐IT

快快快速排序

Rookiep 2023-05-01 原文

排序算法在我们校招考察中很常见,所以今天咋也对很常见的快速排序做出总结,希望能够帮助朋友们理解!

文章目录

一:快速排序

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

二:基本框架

// 假设按照升序对array数组中[left, right)区间中的元素进行排序
void QuickSort(int array[], int left, int right)
{
 	if(right - left <= 1)
 		return;
 
 // 按照基准值对array数组的 [left, right)区间中的元素进行划分
 	int div = partion(array, left, right);
 
 // 划分成功后以div为边界形成了左右两部分 [left, div) 和 [div+1, right)
 // 递归排[left, div)
 	QuickSort(array, left, div);
 
 // 递归排[div+1, right)
 	QuickSort(array, div+1, right);
}

上述为快速排序递归实现的主框架,发现与二叉树前序遍历规则非常像,在写递归框架时可想想二叉树前序遍历规则即可快速写出来。

三:单趟逻辑

选出一个基准值,一般是第一个数或者是最后一个数,经过单趟排序后的结果是左边的值都比基准值小,右边的值都比基准值大。
将区间按照基准值划分为左右两半部分的常见方式有:

3.1:hoare版本

动图演示:

💡💡💡思想:从右边开始找比基准值小的值,从左边找比基准值大的值,等都找到时,交换,再继续上述过程,相遇以后,再把相遇位置的值跟基准值做交换!

int hoare(vector<int>& v, int left, int right)
{
	int keyi = left;
	while (left < right)
	{
		while (left < right && v[right] >= v[keyi])
		{
			right--;
		}
		while (left < right && v[left] <= v[keyi])
		{
			left++;
		}
		swap(v[left], v[right]);
	}
	swap(v[keyi], v[left]);
	return left;
}
void QuickSort(vector<int>& v, int left, int right)
{
	//双闭区间
	if (left >= right)
	{
		return;
	}

	int div = hoare(v, left, right);

	QuickSort(v, left, div - 1);
	QuickSort(v, div + 1, right);
}

💡💡思考:如何保证相遇的位置一定比基准值小呢?

  • 如果选第一个数作为基准值,那么让右边先走,可以保证。
  • 如果选最后一个数作为基准值,那么让左边先走,可以保证。

3.2:挖坑法

先将第一个数据存放在临时变量中,形成一个坑位,比如说找最左边的值作为key,形成一个坑位,那么从右边先走,找比key值更小的值,找到之后把该值放到坑位,形成新的坑位,又从左边找新的比key值大的值放到坑位,依次循环,相遇的位置一定在坑上,再把key值放到坑上。
动图演示:

代码示例:

int DigPit(vector<int>& v, int left, int right)
{
	int key = v[left];
	while (left < right)
	{
		while (left < right && v[right] >= key)
		{
			right--;
		}
		v[left] = v[right];
		while (left < right && v[left] <= key)
		{
			left++;
		}
		v[right] = v[left];
	}
	v[left] = key;
	return left;
}
void QuickSort(vector<int>& v, int left, int right)
{
	//双闭区间
	if (left >= right)
	{
		return;
	}

	int div = DigPit(v, left, right);

	QuickSort(v, left, div - 1);
	QuickSort(v, div + 1, right);
}

相比于版本一的好处在于:

  • 不需要去理解为什么左边做基准值时,要让右边先走。
  • 相遇位置一定在坑位上,再把基准值扔到坑位上。

3.3:前后指针法

初始时,选取第一个值作为基准值,prev指针指向序列开头,cur指针指向prev指针的后一个位置,然后cur指针一直向后找比基准值小的值,找到后,先把prev指针向后移动一位,再交换prev和cur位置的值,最后当cur越界结束,再把prev处的值与序列开头值交换。

int QSPtr(vector<int>& v, int left, int right)
{
	int key = v[left];
	int prev = left;
	int cur = left + 1;
	while (cur <= right)
	{
		if (v[cur] < key && v[prev] != v[cur])
		{
			prev++;
			swap(v[cur], v[prev]);
		}
		cur++;
	}
	swap(v[prev], v[left]);
	return prev;
}
void QuickSort(vector<int>& v, int left, int right)
{
	//双闭区间
	if (left >= right)
	{
		return;
	}

	int div = QSPtr(v, left, right);

	QuickSort(v, left, div - 1);
	QuickSort(v, div + 1, right);
}

注意到prev指针和cur指针中间间隔的是一段比基准值大的区间!

四:快速排序的优化

  • 我们首先思考快速排序的最优情况是什么?

那就是每次选出的key值都大致单趟排序后位于序列中部,那么这时时间复杂度就是n*logn。

  • 其实思考快速排序的最坏情况是什么?
    那就是每次取出的key值都是序列中的最大值或者最小值,此时时间复杂度就是o(n^2)。

所以我们提出一种优化策略,那就是每次取出的数字都是大致是中位数!

💡💡💡首先

  • 三数取中法选key。
int GetmidIndex(vector<int>& v, int left, int right){
	int mid = (left + right) / 2;//left + (right - left) / 2;
	if (v[left] < v[mid])
	{
		if (v[mid] < v[right]){
			return mid;
		}
		else if (v[left] > v[right]){
			return left;
		}
		else{
			return right;
		}
	}
	else{
		if (v[mid] > v[right]){
			return mid;
		}
		else if (v[left] > v[right]){
			return right;
		}
		else{
			return left;
		}
	}
}


int QSPtr(vector<int>& v, int left, int right)
{
	int midi = GetmidIndex(v, left, right);
	swap(v[left], v[midi]);

	//以下步骤都是一样
	int key = v[left];
	int prev = left;
	int cur = left + 1;
	while (cur <= right)
	{
		if (v[cur] < key && v[prev] != v[cur])
		{
			prev++;
			swap(v[cur], v[prev]);
		}
		cur++;
	}
	swap(v[prev], v[left]);
	return prev;
}


void QuickSort(vector<int>& v, int left, int right)
{
	//双闭区间
	if (left >= right)
	{
		return;
	}

	int div = QSPtr(v, left, right);

	QuickSort(v, left, div - 1);
	QuickSort(v, div + 1, right);
}

💡💡💡其次

  • 小区间优化,递归到小的子区间时,可以考虑使用插入排序。
void QuickSort(vector<int>& v, int left, int right)
{
	//双闭区间
	if (left >= right)
	{
		return;
	}
	if (right - left < 10){
		//插入排序
	}

	int div = QSPtr(v, left, right);

	QuickSort(v, left, div - 1);
	QuickSort(v, div + 1, right);
}

当区间很小时,不再使用递归划分的思想让他有序,而是直接使用插入排序对小区间排序,减少递归调用。(我们知道插入排序在小区间大致有序的情况下效率是比较高的)。

五:快速排序的非递归实现(非常重要)

void NonQuickSort(vector<int>& v)
{
	int left = 0, right = v.size() - 1;
	stack<int> st;
	st.push(left);
	st.push(right);

	while (!st.empty())
	{
		int right = st.top();
		st.pop();
		int left = st.top();
		st.pop();

		int div = DigPit(v, left, right);

		if (div - 1 > left)//说明该区间至少两个元素
		{
			st.push(left);
			st.push(div - 1);
		}
		if (right - 1 > div)
		{
			st.push(div + 1);
			st.push(right);
		}
	}
}
int main()
{
	vector<int> v = { 2, 1, 4, 3, 4, 9, 8, 7, 0 };
	NonQuickSort(v);

	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;
	system("pause");
	return 0;
}

六:快速排序的特性总结

  1. 快速排序整体的综合性能和使用场景都是比较好的,所以才敢叫快速排序
  2. 时间复杂度:O(N*logN)
  3. 空间复杂度:O(logN)
  4. 稳定性:不稳定

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

  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一般都是先

随机推荐