归并排序的递归与非递归实现

递归实现

归并排序有点类似于二叉树的后序遍历,是一种基于分治思想的排序算法。具体过程如下:

但要注意,在归并时要额外开辟一个与原数组同等大小的空间用来存储每次归并排序后的值,然后再拷贝到原数组中。

代码实现:

#include<stdlib.h>
#include<string.h>

// 归并排序递归实现
void _MergeSort(int* a, int* tmp, int left, int right)
{
	//当区间只有一个值或没有值时,返回
	if (left >= right)
	{
		return;
	}

	int mid = (left + right) / 2;

	//递归左右区间
	_MergeSort(a, tmp, left, mid);
	_MergeSort(a, tmp, mid + 1, right);

	//归并
	int begin1 = left, end1 = mid;
	int begin2 = mid + 1, end2 = right;

	int i = left;

	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 + left, tmp + left, (right - left + 1) * sizeof(int));
}

void MergeSort(int* a, int n)
{
	//开辟与a同等大小的空间
	int* tmp = (int*)malloc(sizeof(int) * n);

	//实现归并的函数
	_MergeSort(a, tmp, 0, n - 1);

	free(tmp);
	tmp = NULL;
}

非递归实现

在实现快排时,我们用栈来实现非递归,但归并排序时,我们用栈来实现似乎有些麻烦。快排在递归到底时,就已经数组排为有序,但层序遍历不行,层序遍历在递归至最底层时才开始排序,如果要用栈来实现,就需要用两个栈来存储,且过程很麻烦。

因此,在这里我们采用循环的方式来实现层序遍历的非递归。先来看具体过程:

根据上图我们可以得到代码:(但这个代码只能实现2的次方倍的数组个数的排序,其它的会出现数组越界的问题)

// 归并排序非递归实现
void MergeSortNonR(int* a, int n)
{
	//开辟与a同等大小的空间
	int* tmp = (int*)malloc(sizeof(int) * n);

	//归并
	int gap = 1;
	//gap为归并的每组数据的个数
	while (gap < n)
	{
		//i控制每次归并的起始位置的下标
		for (int i = 0; i < n; i += 2 * gap)
		{
			int begin1 = i, end1 = i + gap - 1;
			int begin2 = i + gap, end2 = i + 2 * gap - 1;
			int j = i;

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

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

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

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

			//将数据拷贝到原数组中
			memcpy(a + i, tmp + i, (end2 - i + 1) * sizeof(int));
		}

		gap = 2 * gap;
	}
	
	free(tmp);
	tmp = NULL;
}

要想实现数组归并排序的非递归,我们还要再继续解决数组越界的问题。

先来看越界情况的分析:

代码实现:

// 归并排序非递归实现
void MergeSortNonR(int* a, int n)
{
	//开辟与a同等大小的空间
	int* tmp = (int*)malloc(sizeof(int) * n);

	//归并
	int gap = 1;
	//gap为归并的每组数据的个数
	while (gap < n)
	{
		//i控制每次归并的起始位置的下标
		for (int i = 0; i < n; i += 2 * gap)
		{
			int begin1 = i, end1 = i + gap - 1;
			int begin2 = i + gap, end2 = i + 2 * gap - 1;
			int j = i;

			//结束循环
			if (begin2 >= n)
			{
				break;
			}

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

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

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

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

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

			//将数据拷贝到原数组中
			memcpy(a + i, tmp + i, (end2 - i + 1) * sizeof(int));
		}
		gap = 2 * gap;
	}
	
	free(tmp);
	tmp = NULL;
}

完整代码

#include<stdio.h>
#include<stdlib.h>
#include<string.h>

void Print(int* arr, int n)
{
	for (int i = 0; i < n; i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");
}

// 归并排序递归实现
void _MergeSort(int* a, int* tmp, int left, int right)
{
	//当区间只有一个值或没有值时,返回
	if (left >= right)
	{
		return;
	}

	int mid = (left + right) / 2;

	//递归左右区间
	_MergeSort(a, tmp, left, mid);
	_MergeSort(a, tmp, mid + 1, right);

	//归并
	int begin1 = left, end1 = mid;
	int begin2 = mid + 1, end2 = right;

	int i = left;

	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 + left, tmp + left, (right - left + 1) * sizeof(int));
}

void MergeSort(int* a, int n)
{
	//开辟与a同等大小的空间
	int* tmp = (int*)malloc(sizeof(int) * n);

	//实现归并的函数
	_MergeSort(a, tmp, 0, n - 1);

	free(tmp);
	tmp = NULL;
}

// 归并排序非递归实现
void MergeSortNonR(int* a, int n)
{
	//开辟与a同等大小的空间
	int* tmp = (int*)malloc(sizeof(int) * n);

	//归并
	int gap = 1;
	//gap为归并的每组数据的个数
	while (gap < n)
	{
		//i控制每次归并的起始位置的下标
		for (int i = 0; i < n; i += 2 * gap)
		{
			int begin1 = i, end1 = i + gap - 1;
			int begin2 = i + gap, end2 = i + 2 * gap - 1;
			int j = i;

			//结束循环
			if (begin2 >= n)
			{
				break;
			}

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

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

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

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

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

			//将数据拷贝到原数组中
			memcpy(a + i, tmp + i, (end2 - i + 1) * sizeof(int));
		}
		gap = 2 * gap;
	}
	
	free(tmp);
	tmp = NULL;
}
 
int main()
{
	int arr[] = { 6,5,7,9,2,0,3,1,8,4,10 };
	int len = sizeof(arr) / sizeof(int);
	MergeSortNonR(arr, len);
	Print(arr, len);
	return 0;
}

相关推荐

  1. C语言归并实现

    2024-06-10 16:00:04       12 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-06-10 16:00:04       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-06-10 16:00:04       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-06-10 16:00:04       19 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-06-10 16:00:04       20 阅读

热门阅读

  1. HTML做成一个端午节炫酷页面

    2024-06-10 16:00:04       12 阅读
  2. JUC基础_1.JUC概述&&创建线程的方式

    2024-06-10 16:00:04       8 阅读
  3. C++文件系统

    2024-06-10 16:00:04       8 阅读
  4. 第六章 Three.js 光照

    2024-06-10 16:00:04       8 阅读
  5. ArrayList顺序表简单实现

    2024-06-10 16:00:04       8 阅读
  6. LeetCode 380. Insert Delete GetRandom O(1)

    2024-06-10 16:00:04       8 阅读
  7. [leetcode]first-missing-positive 缺失的第一个正数

    2024-06-10 16:00:04       11 阅读