数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解

这篇具有很好参考价值的文章主要介绍了数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解,数据结构,算法,排序算法,笔记,c语言,开发语言
都说贪小便宜吃大亏,但吃亏是福,那不就是贪小便宜吃大福了吗

一、并归排序 MergeSort

1.基本思想

2.实现原理

3.代码实现

4.归并排序的特性总结

二、非递归并归排序实现

三、完结撒❀

–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀-正文开始-❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–

一、并归排序 MergeSort

1.基本思想

归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。归并排序核心步骤:

数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解,数据结构,算法,排序算法,笔记,c语言,开发语言

2.实现原理

其实先还是运用了递归大事化小的原理,假设我们要对数组arr[i]进行排序,我们可以将该数组均分为两个数组(从中间划分)arr[0] ~ arr[i/2],arr[i/2+1] ~ arr[i],要使arr数组有序可以先将arr[0] ~ arr[i/2]和arr[i/2+1] ~ arr[i]有序,而arr[0] ~ arr[i/2]和arr[i/2+1] ~ arr[i]有序之后可以从头到尾比较两个数组范围内的每一个值,按照要求所排的顺序往新开辟的另一个数组tmp中拷贝,最后再从tmp数组中拷贝到arr数组中完成排序,这就是归并过程。

而要想arr[0] ~ arr[i/2]和arr[i/2+1] ~ arr[i]有序,可以先让arr[0] ~ arr[i/4]和arr[i/4+1]~arr[i/2]有序,arr[i/2+1] ~ arr[mini](中间位置)和arr[mini+1] ~ arr[i]有序,而arr[0] ~ arr[i/4]有序就需要…

这样就可以用递归来实现,数组一直向下递归“分割”,当数组被分为一个数据的时候就可以返回进行合并

所以把并轨排序递归实现按照二叉树来看,完成排序的核心要求就是递归“分割”后,要实现上层数组有序就先要实先下层数组也为有序,完成最后一层数据有序向上回溯即可完成整个数组有序

返回合并过程动态图解:
数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解,数据结构,算法,排序算法,笔记,c语言,开发语言

3.代码实现

void _MergeSort(int* a, int begin, int end, int* tmp)
{
	if (begin == end)
	{
		return;
	}

	int min = (end + begin) / 2;

	_MergeSort(a, begin, min, tmp);
	_MergeSort(a, min+1, end, tmp);

	//并归
	int begin1 = begin, end1 = min;
	int begin2 = min + 1, end2 = end;
	int i = begin;
	while (begin1 <= end1 && begin2 <= end2)
	{
		if (a[begin1] <= a[begin2])
		{
			tmp[i++] = a[begin1++];
		}

		else
		{
			tmp[i++] = a[begin2++];
		}
	}

	while (begin1 <= end1)
	{
		tmp[i++] = a[begin1++];
	}

	while (begin2 <= end2)
	{
		tmp[i++] = a[begin2++];
	}

	memcpy(a+begin, tmp+begin, sizeof(int)*(end - begin + 1));//注意
}

//并归排序 O(N*logN)
void MergeSort(int* a, int n)
{
	assert(a);

	int* tmp = (int*)malloc(sizeof(int) * n);
	if (tmp == NULL)
	{
		perror("malloc fail");
		return;
	}

	_MergeSort(a, 0, n - 1, tmp);

	free(tmp);
	tmp = NULL;
}

4.归并排序的特性总结

1. 归并的缺点在于需要O(N)的空间复杂度,归并排序的思考更多的是解决在磁盘中的外排序问题。

2. 时间复杂度:O(N*logN)

数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解,数据结构,算法,排序算法,笔记,c语言,开发语言

3. 空间复杂度:O(N)

4. 稳定性:稳定

二、非递归并归排序实现

对于归并的非递归实现我们直接利用循环就可以解决。

我们可以将要并归排序的数组直接看成1个数据为一组往下进行并归,并归后再两个数据为一组往下进行并归,直到将数组并归完成为止,见下图:

数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解,数据结构,算法,排序算法,笔记,c语言,开发语言
此过程与递归过程中的回溯过程一样,只是不再需要进行递归分解

这个看起来容易,但实现起来并不简单,我们需要注意许多细节在这里

如何通过一次循环来完成一层数据的并归呢?

以第一层一个数据为一组为例,我们先要完成对数据的控制,开始我们就先要控制10,6一起进行并归,而后再控制7,1进行并归,直到第一层结束为止

完成上述代码实现之后,我们只需在上述代码外再套一层循环来控制gap,即可实现整个代码逻辑

但需要注意的是并不是所有数据最终都会“组队成双”实现归并,如下:

数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解,数据结构,算法,排序算法,笔记,c语言,开发语言
上面数组中最后一组便是一个例子

我们把每次组队并归的两组中,第一组的队头为begin1,队尾为end1,第二组的队头为begin2,队尾为end2
因为begin1不可能越界,那么出现上面问题一共有三种情况:

1.end1越界

当end1越界的话,那么表示这次组队肯定是没有第二组,并且第一组也部分越界,而第一组如果是一个数据的好不用进行并归排序,如果超过一个数据说明在之前已经并归排序完毕,数据也是有序的,所以也不再需要进行并归排序。end1越界直接跳出循环不进行并归排序即可。

2.begin2越界

当begin2越界,情况其实与end1越界一样,不同的可能就是第一组数据都没有越界,没有第二组数据,这时也不需要进行并归排序,直接跳出循环即可。

3.end2越界

当end2越界,表明有第一组数据,第二组数据越界,此时只要将第二组数据的范围缩小到原数组的队尾即可,也就是end2 = n - 1,然后再将两组数据进行并归排序即可(并归排序的实现只要求所给的两组数据有序,不要求数据数量是否相等)。

代码实现:

//并归排序非递归
void MergeSortNonR(int* a, int n)
{
	assert(a);

	int* tmp = (int*)malloc(sizeof(int) * n);
	if (tmp == NULL)
	{
		perror("malloc fail");
		return;
	}

	int gap = 1;

	while (gap < n)
	{
		for (int j = 0; j < n; j+=gap*2)
		{
			int begin1 = j, end1 = begin1 + gap - 1;
			int begin2 = begin1 + gap, end2 = begin2 + gap - 1;

			if (end1 >= n || begin2 >= n)
			{
				break;
			}

			if (end2 >= n)
			{
				end2 = n - 1;
			}

			int i = j;

			while (begin1 <= end1 && begin2 <= end2)
			{
				if (a[begin1] <= a[begin2])
				{
					tmp[i++] = a[begin1++];
				}

				else
				{
					tmp[i++] = a[begin2++];
				}
			}

			while (begin1 <= end1)
			{
				tmp[i++] = a[begin1++];
			}

			while (begin2 <= end2)
			{
				tmp[i++] = a[begin2++];
			}

			memcpy(a + j, tmp + j, sizeof(int) * (end2 - j + 1));
		}
		gap *= 2;
	}

	free(tmp);
	tmp = NULL;
}

三、完结撒❀

如果以上内容对你有帮助不妨点赞支持一下,以后还会分享更多编程知识,我们一起进步。
最后我想讲的是,据说点赞的都能找到漂亮女朋友❤
数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解,数据结构,算法,排序算法,笔记,c语言,开发语言文章来源地址https://www.toymoban.com/news/detail-844823.html

到了这里,关于数据结构进阶篇 之 【并归排序】(递归与非递归实现)详细讲解的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 算法 数据结构 递归插入排序 java插入排序 递归求解插入排序算法 如何用递归写插入排序 插入排序动图 插入排序优化 数据结构(十)

    1. 插入排序(insertion-sort):                                           是一种简单直观的排序算法。它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入     算法稳定性:                  

    2024年02月09日
    浏览(56)
  • 【数据结构】非递归实现快速排序与归并排序

    递归是可以向非递归进行变化的: 比如很经典的 斐波那契数列 可以用 递归 实现也可以用 循环 实现 但是有些复杂的递归仅仅依靠循环是很难控制的, 所以我们需要借助数据结构中的 栈与队列 帮助我们用非递归模拟递归, 故有的时候我们说非递归不是递归却胜似递归 通过

    2024年01月21日
    浏览(51)
  • <数据结构>NO11.归并排序|递归|非递归|优化

    思路: 如果给出两个有序数组,我们很容易可以将它们合并为一个有序数组。 因此当给出一个无序数组时,我们先将它们均分为两组有序数组,在将这两组有序数组合并为一个有序数组;而将原数组分成2组 有序数组的思路又是归并排序 。 递归的结束条件是什么? 当递归区间

    2024年02月16日
    浏览(43)
  • 数据结构排序——详细讲解归并排序(c语言实现递归及非递归)

    上次是快排和冒泡:数据结构排序——详解快排及其优化和冒泡排序(c语言实现、附有图片与动图示意) 今天为大家带来归并排序 归并排序是一种分治算法,它将序列分成两个子序列,分别对 子序列进行排序 ,然后将排序好的子序列 合并起来 。这个过程可以 递归 地进行,

    2024年01月22日
    浏览(67)
  • 【数据结构与算法】:非递归实现快速排序、归并排序

    🔥 个人主页 : Quitecoder 🔥 专栏 :数据结构与算法 上篇文章我们详细讲解了递归版本的快速排序,本篇我们来探究非递归实现快速排序和归并排序 快速排序的非递归实现主要依赖于栈(stack)来模拟递归过程中的函数调用栈。递归版本的快速排序通过递归调用自身来处理子

    2024年03月24日
    浏览(55)
  • 【数据结构】手撕归并排序(含非递归)

    目录 一,归并排序(递归) 1,基本思想  2,思路实现 二,归并排序(非递归) 1,思路实现 2,归并排序的特性总结: 1,基本思想 归并排序 (MERGE-SORT) 是建立在 归并操作 上的一种 有效的排序算法 ,该算法是采用 分治法(Divide and Conquer) 的一个非常典型的应用; 将 已

    2024年02月08日
    浏览(47)
  • [数据结构 -- 手撕排序算法第七篇] 递归实现归并排序

    目录 1、归并的思想 2、归并排序的思想 2.1 基本思想 2.2 图解分析 3、归并排序递归版本代码实现 3.1 代码解析 3.2 注意事项 3.2.1错误划分:[begin, mid-1],[mid, end] 3.2.2 正确划分:[begin, mid], [mid+1, end] 4、归并排序的测试 5、时间复杂度、空间复杂度分析 5.1 时间复杂度 5.2 空间复杂

    2024年02月16日
    浏览(50)
  • 【数据结构与算法】归并排序详解:归并排序算法,归并排序非递归实现

    归并排序是一种经典的排序算法,它使用了分治法的思想。下面是归并排序的算法思想: 递归地将数组划分成较小的子数组,直到每个子数组的长度为1或者0。 将相邻的子数组合并,形成更大的已排序的数组,直到最终得到一个完全排序的数组。 归并排序的过程可以分为三

    2024年01月22日
    浏览(70)
  • 【数据结构】论如何拿捏快速排序?(含非递归)

    目录 一,快速排序(递归) 1,快排思想 2,霍尔排序 3,挖坑法 4,前后指针法 5,快速排序优化 1,三数取中法选key 2,小区间优化 二,快速排序(非递归) Stack.h Stack.c 三,快速排序源代码 1,快排思想 快速排序是 Hoare于1962年 提出的一种 二叉树结构的交换 排序方法,其

    2024年02月08日
    浏览(58)
  • 【数据结构与算法】快速排序的非递归实现方法

      目录 一.前言 二.非递归实现 如果数据量过大的话,不断递归就会出现 栈溢出 的现象,这个时候你的代码是没问题的,但就是跑不起来,这个时候就要 把递归改成非递归 。 一般有两种改法: 1.直接改,利用循环等; 2.借助栈的辅助。 而快速排序的非递归实现方法就需要

    2023年04月17日
    浏览(55)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包