C语言算法进阶——快速排序,归并排序与递归


前言

快速排序与归并排序是递归的两个典型例子:
快排是在“递”的过程中将数组排好序,“归”的过程中不做任何操作;
归并排序是在“递”的过程中不做任何操作,在“归”的过程将数组排好序。


一、快速排序

思路:在每一层中找到a[left]的位置,小于a[left]在他的左边,大于a[left]在他的右边,然后递归排序以a[left]当前位置的左右数组,直到将所以元素放在它合适的位置。

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
void quick_sort(int *a,int left,int right)
{
	if(left<right)//递归的结束条件
	{
		int temp=a[left],lab=0;//temp为当前层的需要确定位置的元素,lab是一个标识符
		int l=left,r=right;//定义双指针
		while(l<r)//使用双指针找到temp的位置
		{
			if(lab==0)//此时从数组末尾往回比较
			{
				if(temp>a[r])//找到比temp小的数交换位置
				{
					a[l]=a[r];
					a[r]=temp;
					lab=1;//标识符置1
				}
				else
					r--;//没找到则继续向前找
			}
			if(lab==1)//由于在数组末尾找到比temp小的数,temp已经交换位置,此时从数组开头向后找
			{
				if(temp<a[l])//找到比temp大的数则交换位置
				{
					a[r]=a[l];
					a[l]=temp;
					lab=0;//标识符置0
				}
				else
					l++;//没找到则继续向后找
			}
	}
	quick_sort(a,left,l-1);//递归temp以左的数组
	quick_sort(a,r+1,right);//递归temp以右的数组
	}
}
int main()
{
	int i;
	int a[]={50,36,66,76,36,12,25,95,78,13,89,100};//随便找了组数测试一下
	quick_sort(a,0,11);
	for(i=0;i<sizeof(a)/sizeof(a[0]);i++)
	{
		printf("%d,",a[i]);
	}
	printf("\n");
}

结果展示

在这里插入图片描述

二、归并排序

思路:归并的分治思想有些像树
1、在的过程中把数组分解成[left,mid]和[mid+1,right]两部分,在的第二层,把[left,mid]看成一个整体又分为[left,mid]和[mid+1,right]两部分,[mid+1,right]同理,分解到最后把一个整体分成了若干个叶子节点。
2、在的过程中,叶节点2,5比较大小后返回到上一层得到排好序的父节点2,5,父节点再与他的兄弟节点比较,向上层返回排好序的节点,最终整个数组有序。
分解的过程:
在这里插入图片描述

#include<stdio.h>
#include<stdlib.h>
void _MergeSort(int* arr, int left, int right,int* tmp)
{
	if (left >= right)//=为只有一个数,>为区间存在
		return;
	int mid = (left + right) >> 1;//左右区间如果没有序,分治递归分割区间,直到最小,
	_MergeSort(arr, left, mid, tmp);//区间被分为[left,mid] 和 [mid+1,right],开始递归
	_MergeSort(arr, mid + 1, right, tmp);
	int i;
	//此时通过递归已经能保证左右子区间有序
	//开始归并
	int begin1 = left;
	int end1 = mid;
	int begin2 = mid + 1;
	int end2 = right;
	int index = left;	//归并时放入临时数组的位置从left开始
	while (begin1 <= end1 && begin2 <= end2)
	{
		if (arr[begin1] < arr[begin2])
			tmp[index++] = arr[begin1++];
		else
			tmp[index++] = arr[begin2++];
	}
	while (begin1 <= end1)	//循环结束,将还没遍历完的那个区间剩下的数拷贝下来
		tmp[index++] = arr[begin1++];
	while (begin2 <= end2)
		tmp[index++] = arr[begin2++];
	for(i = left; i <=right ; i++)//将排归并完的数拷贝回原数组
		arr[i] = tmp[i];
}
void MergeSort(int* arr, int n)
{ 
	int* tmp = (int*)malloc(sizeof(int)*n);	//申请一个空间用来临时存放数据
	_MergeSort(arr, 0, n - 1, tmp);//归并排序
	free(tmp);//释放空间
}
int main()
{
	int len,i;
	printf("请输入数组长度:");
	scanf("%d",&len);
	int *arr=malloc(len*sizeof(int));
	for(i=0;i<len;i++)
	{
		printf("请输入数组第%d位的值:",i);
		scanf("%d",arr+i);
	}
	MergeSort(arr,len);
	for(i=0;i<len;i++)
	{
		printf("数组第%d位的值为%d\n",i,arr[i]);
	}
}

结果展示

在这里插入图片描述

三、总结

1、无论是快速排序还是归并排序理解起来都是有一定的难度的,很容易陷入一些误区中,产生一些偏离正确思路的想法,但是多看多想,一定会回归正途的;
2、在这两个排序实现的代码中,我们可以看到,快速排序是在递归前实现一些功能,归并排序是在回溯的过程中实现一些功能,这也是递归一个特点,你可以选择在的过程处理,也可以在的过程处理;
3、看完以上内容你是否对递归了解更深一些?当然,如果你没什么体会,那就怪我说的不明白。如果你觉得我写的有些小儿科,大神,请收下我的膝盖!

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值