当前位置: 首页 > article >正文

每日回顾:简单用C写 归并排序

归并排序

归并排序(Merge Sort)是一种分治算法(Divide and Conquer),它将数组分成两半,分别对它们进行排序,然后将排序好的两半合并在一起。

递归版本

避免重复 malloc 空间,定义子函数来执行递归,大致思想:类似于二叉树的后根遍历,递归的对左子树和右子树进行归并,下面示意图展示第一次归并过程;归并排序需要开辟额外的数组空间,用以存放每次归并操作后的结果。

其他排序基本只能应用于内存上的排序,而归并排序可以解决在磁盘中的外排序问题。

//递归版本子函数
void _MergeSort(int* a, int begin, int end, int* tmp) {
	//递归出口
	if (begin == end) {
		return;
	}

	int mid = (begin + end) / 2;
	// [begin,mid] [mid+1,end]
	_MergeSort(a, begin, mid, tmp);
	_MergeSort(a, mid+1, end, tmp);

	//至此,将左右区间中的共 2 个及以上的元素进行归并
	int begin1 = begin, end1 = mid;
	int begin2 = mid + 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++];
		}
	}
	//至此,需要归并的2个区间,已完成一个,处理另一个
	while (begin1 <= end1) {
		tmp[i++] = a[begin1++];
	}
	while (begin2 <= end2) {
		tmp[i++] = a[begin2++];
	}

	//每次归并都拷贝到原数组一遍,指针要加上 begin
	memcpy(a+begin, tmp+begin, sizeof(int) * (end - begin + 1));
}

//归并排序_递归
void MergeSort(int* a, int begin, int end) {
	//区间不存在或只有一个元素
	if (begin >= end) {
		return;
	}

	int* tmp = (int*)malloc(sizeof(int) * (end - begin + 1));
	if (tmp == NULL) {
		perror("malloc failed");
		return;
	}
	
	_MergeSort(a, begin, end, tmp);

	free(tmp);
}

非递归版本

大致思想:非递归版本中使用局部变量 gap ,进行分组,然后两两归并

需要注意,如果数组元素数量不是每组中 gap 的倍数,那么肯定会出现越界访问的情况,所以需要对越界的区间访问进行修正

//归并排序_非递归
void MergeSortNonR(int* a, int n) {
	int* tmp = (int*)malloc(sizeof(int) * n);

	int gap = 1;
	while (gap < n) {
		int j = 0;
		for (int i = 0; i < n; i += 2 * gap) {

			//仍然是每 2 组归并
			int begin1 = i, end1 = i + gap - 1;
			int begin2 = i + gap, end2 = i + gap * 2 - 1;

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

			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, sizeof(int) * (end2 - i + 1));
			gap *= 2;
		}
	}
	free(tmp);
}

http://www.kler.cn/news/364544.html

相关文章:

  • 盘古信息制造数字化优才计划 | 解决人才困境 赋能智能制造
  • 开源模型应用落地-Qwen2.5-7B-Instruct与vllm实现推理加速的正确姿势-Gradio
  • 【OpenAI】第六节(语音生成与语音识别技术)从 ChatGPT 到 Whisper 的全方位指南
  • Windows通过netsh控制安全中心防火墙和网络保护策略
  • 【mysql进阶】2-4. mysql 系统库
  • 【景观生态学实验】实验二 景观类型分类
  • 光通信——前传基本架构
  • Next.js14快速上手
  • spark sql 广播模式参数
  • 二叉树的性质
  • 基于Springboot在线视频网站的设计与实现
  • 深入解析东芝TB62261FTG,步进电机驱动方案
  • python之数据结构与算法(数据结构篇)—— 线性表
  • 笛卡尔空间内的阻抗控制
  • DAY62WEB 攻防-PHP 反序列化CLI 框架类PHPGGC 生成器TPYiiLaravel 等利用
  • openresty安装
  • 【再谈设计模式】工厂模式~制造者的艺术
  • tomcat基本配置
  • 高性能数据分析利器DuckDB在Python中的使用
  • Web页面测试方法「详细介绍」
  • 【赵渝强老师】Oracle的控制文件与归档日志文件
  • python:pygame, pyOpenGL 示例:旋转的八面体
  • JAVA 单例模式实验(头歌)
  • 【ROS GitHub使用】
  • ​8.13TB高清卫星影像更新(WGS84坐标投影)
  • 简单三步完成 Telegram 生态的 Web3 冷启动