算法之归并排序

news2024/11/24 16:29:12

文章目录

  • 一、归并排序(递归版)
  • 二、归并排序(非递归版)


一、归并排序(递归版)

归并排序思想:将数组划分为两个区间,左区间,右区间 然后对这两个区间内容进行排序 ,这两个区间排好序之后再将其合并为一个有序的区间
在这里插入图片描述

这两个区间排好序之后,再将这两个区间合并为一个区间 也就是将这两个区间的数据排序为一个有序的区间
而将数组划分为两个区间之后是如何将这两个区间里的内容排好序的呢 是重复同样的操 作再将这两个区间中的左区间分别又划分为两个区间(左区间,右区间)在这里插入图片描述
,将这两个区间分别排好序之后,
再归为一个区间,也就是左区间有序了,然后再排它的右区间,此时它的右区间右划分为两个区间(左区间,右区间)在这里插入图片描述
对它们分别进行排序 ,而划分下去的左右区间又要执行同样的操作,直到最后区间大小为一时,那么此时就不用排返回,返回时与另一个区间比较进行排序,
最后右区间变为有序的了,最后将这两个大的左右区间归并排序为一个区间,此时这个区间有序 ,由于递归排序回来时,将小区间排好序,最后整个大区间跟着有序了在这里插入图片描述
我上图只是画了整个数组左区间的,右区间可以下去尝试下,右区间也是如此

void _MergeSort(int* a, int left, int right, int* tmp)
{
	//当区间中只有一个数时就不用再对其进行排序
	if (left >= right)
		return;

	//每次将其区间折半划分为左区间和右区间
	int mid = (left + right) / 2;
	//再重复划分,直到它的左右区间只剩下一个数,那么再返回
	//对其排序
	_MergeSort(a, left, mid, tmp);
	_MergeSort(a, mid + 1, right, tmp);

	//将其小区间排好序后再排它的大区间
	//小区间有序之后大区间排过来也就是有序的了
	int begin1 = left, end1 = mid;
	int begin2 = mid + 1, end2 = right;
	int begin = left;

	//将左右区间合并为一个区间
	while (begin1 <= end1 && begin2 <= end2)
	{
		if (a[begin1] > a[begin2])
		{
			tmp[begin++] = a[begin2++];
		}

		else
		{
			tmp[begin++] = a[begin1++];
		}
	}
    
    //两个区间中还剩余数据的那个区间直接尾插到tmp数组后面
	while (begin1 <= end1)
	{
		tmp[begin++] = a[begin1++];
	}

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

	//最后将排好序的区间拷贝回原来的数组
	memcpy(a + left, tmp+left, sizeof(int)*(right - left + 1));
}

void MergeSort(int* a, int n)
{
	//开一个临时数组用来存放小区间排序后的数据
	//最后再将排序后的数组拷贝回原数组中
	int* tmp = (int*)malloc(sizeof(int) * n);
	if (tmp == NULL)
	{
		perror("malloc fail\n");
		return;
	}

	//归并排序分两区间进行
	_MergeSort(a, 0, n - 1, tmp);
	free(tmp);
}

二、归并排序(非递归版)

思想:将整个数组一个数与一个数比较,然后将它们排序为一个有两个数的区间 得到两个数与两个数之间有序
再将有两个数的区间与另外有两个数的区间进行排序为一个有四个数的区间 得到四个数的区间与四个数的区间有序
再将四个数的有序区间与四个数的有序区间进行排序为一个有8个数的有序区间 得到8个数与8个数之间有序
如此循环下去,直到区间中数据的个数为原数组数据个数时为止
那么区间中的数据个数如何控制,它又是如何变化的,设定一个变量gap来控制,然后当数组里面的数据都两两归并后gap变为原来2倍
gap *= 2
在这里插入图片描述

当数组长度不是gap倍数时,要注意控制边界。

那这个边界如何控制?

设前一个区间左端:begin1,右端:end1
设后一个区间左端为:begin2,右端:end2
如果第一个区间的右端已经超出了数组长度(end1>=n),此时将第一个区间右端下标重定为数组长度-1(end1 =n-1, begin2 = n, end2 = n-1),如果前一个区间右端没有超出,但是第二个区间左端就超出了(重定begin2 = n-1,end2 = n-1,),这样后一个区间就不用再与前一个区间合并了,直接将前一个区间拷到原数组后面,如果后一个区间左端没有超出,但是后一个区间右端超出了数组长度,此时重定end2,end2 = n-1,此时前一个再与后一个合并

** 将tmp中内容拷贝回原数组**
在这里插入图片描述

将间隔为gap的小区间归并后的内容整体拷贝回原数组

void MergeSort1(int* a, int n)
{
	int* tmp = (int*)malloc(sizeof(int) * n);
	if (tmp == NULL)
	{
		perror("malloc fail\n");
		return;
	}

	int gap = 1;
	while (gap < n)
	{
	     //小区间由gap控制,且每次都是前一个区间和后一个区间
	     //归并为一个大的有序区间,所以每次i需要加上gap的2倍,
	     //一次跳过两个区间
		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;//由于两个小区间要合并为一个区间,
			//然后存放到数组tmp中,合并后数组tmp长度大小
			//为两个小区间长度大小之和,然后下次在开始合并时,
			//从合并后的右端下标开始赋值,所以tmp下标起始由i
			//决定
			if (end1 >= n)
			{
				end1 = n - 1;
				begin2 = n ;
				end2 = n-1;
			}

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

			else if(end2 >=n )
			{
				end2 = n - 1;
			}
             //两个区间中小的数尾插到tmp数组中
			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++];
			}

			
		}
		//将间隔为gap小区间合并后的数组tmp里的内容全部拷贝回原数组中
        memcpy(a, tmp, sizeof(int) * n);
		gap *= 2;//最后gap变为原来的2倍
		
	}
	
	free(tmp);
}

将间隔为gap的小区间一段一段的拷贝回原数组

void MergeSort2(int* a, int n)
{
	int* tmp = (int*)malloc(sizeof(int) * n);
	if (tmp == NULL)
	{
		perror("malloc fail\n");
		return;
	}

	int gap = 1;
	while (gap < n)
	{
		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;
            //由于gap使其分区间块化有序,此时前一个区间的右端
            //或者后一个区间的左端超出数组长度了,那么直接
            //拷贝到原数组后面
			if (end1 >= n || begin1 >= n)
			{
				break;
			}
			else 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++];
			}
			//每一次区间间隔为gap的两个区间合并后就将其拷贝到
			//原数组中
			memcpy(a + i, tmp + i, sizeof(int) * (end2 - i + 1));
		}

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



归并排序将两个有序的区间归并为一个有序的区间

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/426203.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

一文讲懂C#、ASP.NET、ASP.NET MVC、ASP.NET web form、asp.net core mvc的区别

微软的命名很糟糕。技术上有两个框架&#xff1a;ASP.NET和ASP.NET Core&#xff0c;它们分别基于.NET Framework和.NET Core构建。 当Microsoft首次尝试创建一个遵循MVC模式的“现代”Web应用程序平台时&#xff0c;它将这个新平台称为“ASP.NET MVC”&#xff0c;以区别于以前…

【性能优化】cocoscreator 共享节点-动效复用方案

前言 迷雾散尽&#xff0c;露出了古朴庄严的森林。古老的铁杉&#xff0c;在头顶编成绿色穹顶。 阳光在树叶间破碎成金色顶棚。从树干间远眺&#xff0c;远处的森林渐渐隐去。 用几句话就能描述一片巨大的森林&#xff0c;但是在实时游戏中做这件事就完全是另外一件事了。 当屏…

[数据结构-C语言] 算法的时间复杂度

目录 1.算法的复杂度 2.时间复杂度 2.1 时间复杂度的概念 2.2 大O的渐进表示法 3、常见时间复杂度计算举例 3.1 冒泡排序 3.2 二分查找 3.3 阶乘递归 3.4 斐波那契数列 1.算法的复杂度 算法在编写成可执行程序后&#xff0c;运行时需要耗费时间资源和空间(内存)资源 …

PS学习记录——自动选择的【图层】/【组】模式说明

PS中&#xff0c;【移动工具】状态下&#xff0c;上方的工具栏有个【自动选择功能】&#xff0c;用于移动图像 自动选择下有【图层】、【组】两种选项 1、自动选择-【图层】 图层模式下&#xff0c;自动选择可以任意移动选中的图像图层&#xff0c;想移动哪个就移动哪个 即便不…

43.CSS grid布局

本节我们学习的初始代码如下&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta http-equiv"X-UA-Compatible" content"IEedge" /><meta name"viewport" …

110页智慧农业解决方案(农业信息化解决方案)(ppt可编辑)

本资料来源公开网络&#xff0c;仅供个人学习&#xff0c;请勿商用&#xff0c;如有侵权请联系删除。 第一部分 智慧农业概述 智慧农业以农业资源为基础、市场为导向、效益为中心、产业化为抓手&#xff0c;面向农业管理部门、农技推广部门、农业企业、农业园区和基地、农业专家…

【C】Pointer

系列连载 【python / C / C】 参考 《C语言程序设计&#xff08;第四版&#xff09;谭浩强》【C语言】C语言视频教程《郝斌 C 语言自学教程》 文章目录一、指针的重要性二、指针的定义三、指针的分类基本类型指针指针和一维数组指针和函数指针和结构体多级指针int * p;// p …

CASS自带数据-绘制地形图

1、打开cass软件&#xff0c;如下图&#xff1a; 2、点击“绘图处理”菜单栏&#xff0c;选择“展野外测点点号”&#xff0c;如下&#xff1a; 3、在命令栏输入绘图比例尺&#xff0c;这里选择默认1:500&#xff0c;点击回车键进入下一步&#xff0c; 4、在弹出的窗口中&#…

单片机通过串口向电脑端发送数据电脑端发送数据控制led

上节课我们学习了串口的理论部分&#xff0c;这节课我们要来学习实操部分。 要想实现单片机通过串口向电脑端发送数据&#xff0c;我们首先要来配置寄存器。 1.配置SCON SCON寄存器中的SM0配置为0&#xff0c;SM1配置为1决定了串口工作在模式一&#xff0c;也就是8位UART&am…

前端透明分栏设计

目前在开发一个基于众包的打分网站&#xff0c;前端遇到的一个需求是&#xff1a; 背景是电影的海报&#xff0c;且不能随着scroll-bar滚动&#xff0c;需要一个蒙版分为两栏&#xff0c;左侧是影视的媒体信息&#xff0c;不随页面滚动右侧是影视的基本信息和评分信息&#xf…

如何使用基于GPT-4的Cursor编辑器提升开发效率

程序员最恨两件事情&#xff1a;一是别人代码不写文档&#xff0c;二是要让自己写文档。随着 GPT-4 的到来这些都不是问题了&#xff0c;顺带可能连程序员都解决了。。。 之前一直觉得 AI 生成的代码也就写个面试题的水平&#xff0c;小打小闹&#xff0c;现在时代可变了。Curs…

@Conditional四个较常用的派生注解总结

该内容是在学习SpringBoot底层的时候&#xff0c;看到SpringBoot的使用&#xff0c;所以这的做一个简单的总结方便以后复习。可能会有一些问题&#xff0c;还望指出共同学习 Conditional注解&#xff1a; 作用&#xff1a;按照一定的条件进行判断&#xff0c;在满足给定条件后…

如果采用密钥对的形式登录系统后,如何由普通用户切换到root用户

使用xshell工具 采用秘钥的方式进行登录 创建一个新用户并设置密码 切换到新用户之后 su root 想要切换成root用户的时候 输入密码提示 su:Authentication failure su&#xff1a;身份验证失败 使用秘钥之后是不能使用密码再进行登录 我去阿里云查了一下 有一个相同…

stable diffusion 安装xFormers 报错:Couldn‘t install open_clip.

一、No module ‘xformers’. Proceeding without it. 这是因为没有安装xformers导致的。 解决办法&#xff1a; 在webui-user.bat文件这添加一行&#xff1a; set COMMANDLINE_ARGS--xformers如下图所示&#xff1a; 试着点击webui-user.bat&#xff0c;看能否下载&#xff…

17:00面试,17:04就出来了 ,问的实在是太...

从外包出来&#xff0c;没想到算法死在另一家厂子 自从加入这家公司&#xff0c;每天都在加班&#xff0c;钱倒是给的不少&#xff0c;所以也就忍了。没想到8月一纸通知&#xff0c;所有人不许加班&#xff0c;薪资直降30%&#xff0c;顿时有吃不起饭的赶脚。 好在有个兄弟内推…

你具备抽离与封装的思想吗?

笔者最近思考了自己参与的分布式系统业务的架构小细节&#xff0c;虽然笔者每天做的是实现部分需求与业务&#xff0c;但是笔者还是拥有很多时间去读底层源码的&#xff0c;加之笔者自身的思考与实践demo的总结&#xff0c;笔者将在本篇文章中提出笔者自己对“抽离”与“封装”…

Nginx编译安装及配置文件详解

写在前面 Centos版本&#xff1a;Centos 7.6 - 64bit Nginx版本&#xff1a;1.20.2 一、什么是Nginx Nginx (engine x) 是一款轻量级的Web 服务器 、反向代理服务器及电子邮件&#xff08;IMAP/POP3&#xff09;代理服务器。 二、Nginx用在哪些地方 2.1 静态资源服务 动静…

mkv视频文件怎么转成mp4,这3个方法很好用

对于那些电脑不是很精通的小伙伴来说&#xff0c; mkv视频文件怎么转成mp4&#xff1f;小编先来介绍一下mkv视频格式&#xff0c;mkv是一种开放标准的自由的容器和文件格式&#xff0c;是一种多媒体封装格式&#xff0c;能够在一个文件中容纳无限数量的视频、音频、图片或字幕轨…

MVCC底层原理

目录说明MVCC的底层原理隐藏字段undo logRead View说明 在被面试官问面试题的时候&#xff0c;首先它问了Mysql的事务的隔离级别有几种&#xff1f;默认是哪种&#xff1f;他们分别解决了什么问题&#xff1f; 我在一顿回答“巴巴巴巴。。。。”之后&#xff0c;它又继续问题…

追踪项目进展常用的衡量方法

作为项目管理活动&#xff0c;项目中的进度跟踪可以帮助你了解项目实时的完成状态。有了正确的方法、项目管理工具和指标&#xff0c;项目跟踪可以指导你的团队的行动&#xff0c;甚至在问题出现之前向你发出预警。 项目经理使用各种方法、工具和指标来跟踪进度。其中最常见的…