快速排序-hoare版、挖坑法、前后指针法

这篇博客详细介绍了快速排序的三种不同实现方法:hoare版(左右指针法)、挖坑法和前后指针法。在递归和非递归场景下,通过代码展示了如何进行排序操作。通过对基准值的比较和交换,实现序列的有效排序。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

快速排序基本可以分为两种:递归和非递归法。又可以将区间按照基准值划分为左右两半部分进行排序,排序方法有hoare版(左右指针法)、挖坑法、前后指针法。好话不多说,上代码。

递归

快排的递归方法

```

void Quicksort(int *arr, int left, int right)
{
	if (right - left > 1)
	{
		int div = Partion(arr, left, right);
		Quicksort(arr, left, div-1);
		Quicksort(arr, div+1, right);
	}
}


```

hoare版(左右指针法)


首先选一个key为基准,建立一个begin=left、end=right,如果begin小于key,则begin++,如果end大于key,则end--,当begin大于key的值,而end小于key的值时,对两个数的位置进行交换,当begin遇到end时,将基准值和begin进行交换

int Partion(int* arr, int left, int right)
{
	int mid = GetMidIndex(arr, left, right);
	if (mid != right)
		swap(&arr[mid], &arr[right]);
	int begin = left;
	int end = right;
	int key = arr[right];
	while (begin < end)
	{
		while (begin < end && arr[begin] <= key)
			begin++;
		while (begin < end && arr[end] >= key)
			end--;
		if (begin < end)
		{
			swap(&arr[begin], &arr[end]);
		}

	}
	swap(&arr[right], &arr[begin]);
	return begin;
}

挖坑法

 

 


  首先选取一个key值,设置两个变量begin = left ,end = right,从left一直向后走,直到找到一个大于key的值,然后将该数放入坑中,坑位变成了array[begin],right一直向前走,直到找到一个小于key的值,然后将该数放入坑中,坑位变成了array[end]。
重复3和4的步骤,直到left和right相遇,然后将key放入最后一个坑位。

 

int Partion2(int* arr, int left, int right)
{
	int mid = GetMidIndex(arr, left, right);
	if (mid != right)
		swap(&arr[mid], &arr[right]);
	int begin = left;
	int end = right;
	int key = arr[right];
	while (begin < end)
	{
		while (begin < end && arr[begin] <= key)
			begin++;
		arr[end] = arr[begin];
		while (begin < end && arr[end] >= key)
			end--;
		arr[begin] = arr[end];
	}
	arr[end] = key;
	return begin;
}

前后指针法

定义变量cur指向序列的开头,定义变量pre指向cur的前一个位置。当array[cur]  < key时,cur和pre同时往后走,如果array[cur]>key,cur往后走,pre留在大于key的数值前一个位置。当array[cur]再次 < key时,交换array[cur]和array[pre]。

在没找到大于key值前,pre永远紧跟cur,遇到大的两者之间机会拉开差距,中间差的肯定是连续的大于key的值,当再次遇到小于key的值时,交换两个下标对应的值就好了

int Partion3(int* arr, int left, int right)
{
	int mid = GetMidIndex(arr,left,right);
	if (mid != right)
	   swap(&arr[mid], &arr[right]);
	int key = arr[right];
	int cur = left;
	int pre = cur - 1;
	while (cur < right){
		if(arr[cur] <= key && ++pre != cur){
			swap(&arr[cur], &arr[pre]);
		}
		++cur;
    }
	if (++pre != cur && pre != right)
		swap(&arr[pre], &arr[right]);
	return pre;
}

非递归

递归转化为非递归就是压栈

void QuickSortNor(int *arr, int left, int right)
{
	stack<int> s;
	s.push(left);
	s.push(right);
	while (s.empty()){
		int right = s.top();
		s.pop();
		int left = s.top();
		s.pop();
		int Index = Partion(arr, left, right);
		if (Index - 1 > left)
		{
			s.push(left);
			s.push(Index - 1);
		}
		if (Index + 1 < right)
		{
			s.push(Index + 1);
			s.push(right);
		}

	}
}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 


 

 

 

回答: 快速排序Hoare是一种常用的快速排序实现方。它是由Tony Hoare在1960年提出的。Hoare的基本思想是选择一个基准值,将待排序序列分成两部分,一部分是小于基准值的元素,另一部分是大于基准值的元素。然后对这两部分分别进行递归排序,最终将整个序列排序完成。具体实现过程如下: 1. 选择一个基准值,可以是序列中的任意一个元素。 2. 定义两个指针,一个指向序列的起始位置,一个指向序列的末尾位置。 3. 移动左指针,直到找到一个大于等于基准值的元素。 4. 移动右指针,直到找到一个小于等于基准值的元素。 5. 如果左指针小于等于右指针,则交换左右指针所指向的元素。 6. 继续移动左右指针,直到左指针大于右指针。 7. 将基准值与右指针所指向的元素交换。 8. 分别对基准值左边和右边的子序列进行递归排序。 通过以上步骤,每一次递归都会将基准值放置在正确的位置上,最终完成整个序列的排序。\[1\]\[2\]\[3\] #### 引用[.reference_title] - *1* *2* [【八大排序③】快速排序(动图演绎Hoare挖坑前后指针法)](https://blog.youkuaiyun.com/Living_Amethyst/article/details/125513838)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [快速排序常见3种方(hoare挖坑前后指针法)以及改进。](https://blog.youkuaiyun.com/tjh1998/article/details/122159488)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值