【算法与数据结构】归并排序的代码实现(详细图解)以及master公式的讲解

news2024/9/20 16:39:30

目录

1、归并排序

 1.1、算法描述

 1.2、图解说明

2、代码实现 

3、master公式

3.1、公式以及结论

3.2、适用于某些特殊的递归

3.3、计算归并排序的时间复杂度


1、归并排序

归并排序是建立在归并操作上的一种有效的排序算法。该算法是采用递归或者说是分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。

若将两个有序表合并成一个有序表,称为二路归并

 1.1、算法描述

  • 把长度为n的输入序列分成两个长度为n/2的子序列;
  • 对这两个子序列分别采用归并排序;
  • 将两个排序好的子序列合并成一个最终的排序序列。

而将两个的有序数列合并成一个有序数列,我们称之为"归并",这就是归并排序名字的由来。

 1.2、图解说明

一句话简单说:对L到R范围排序,可以先求出L到R的中点M。先让左侧数据排好序,然后再让右侧数据排好序,此时再将两个有序子序列整合成一个新的有序序列。

 例如:

对一个数组[8,3,6,4,2,1,5,7]进行归并排序。

第一步:把长度为n的输入序列分成两个长度为n/2的子序列,新的子序列再分别分成两个长度为自身一半也就是n/4的子序列,以此类推。当分到单个子序列只剩下一个数字时,一个数字就是天然了有序,即此时左侧和右侧都排好序了。

第二步:将两个排序好的子序列合并成一个新的排序序列。首先在每个子序列中都有一个指针指向子序列的第一个元素,两个指针的元素两两比较,较小的元素先放入新的子序列中,然后指针挪动继续比较,直至全部放入新的子序列当中,即完成一次子序列合并。慢慢合并最终使所有元素都成有序,即完成归并排序。

 这个思路过程是非常精髓的,理解了这个思路之后,就可以试着用代码实现了。

2、代码实现 

要使用归并,首先需要知道数组arr以及数组最左L下标和最右R下标,因此需要求出并带入MergeSort当中。

int main()
{
	int arr[10] = { 8,3,6,4,2,1,5,7 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	MergeSort(arr, 0, sz - 1);
	int i = 0;
	for ( i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

接下来看看MergeSort的实现。

  1. 首先是判断L是否等于R,如果L和R相等时就相当于是单个子序列中只存在了一个元素,而此时该子序列就为有序,因此不用进行操作直接返回即可,即if(L == R) return;
  2. 如果L不等于R时则认为该子序列中仍然可拆分,便求出mid中间值,并分别进行两次递归让两块递归范围有序,递归范围是L到mid、mid+1到R。
  3. 最后当递归结束时则代表L到mid、mid+1到R序列已有序,整体还无序,此时需要使用ExternalSort外部排序将这两个子序列整合成一个新的有序序列。
void MergeSort(int* arr, int L, int R)
{
	if (L == R)  //子序列只有一个数,默认为有序
	{
		return;
	}
	int mid = L + (R - L) / 2;
	MergeSort(arr, L, mid);
	MergeSort(arr, mid + 1, R);
	ExternalSort(arr, L, mid, R);
}

ExternalSort的作用就是让arr中的L到M、M+1到R合并成一个新的有序序列,并将判断后的结果序列先存入到help指针指向的区域,等待完成所有合并,再将help整个区域的数据拷贝到arr对应的位置。

而存入help的规则是:

1、如果p1和p2都没有越界进行while循环:

p1和p2比较,如果p1大于p2,则将p2所指向的元素放入help中,然后将p2右移指向下一个,继续下一轮比较。

p1和p2比较,如果p1小于等于p2,则将p1所指向的元素放入help中,然后将p1右移指向下一个,继续下一轮比较。

2、如果有一方越界了,则退出循环,并判断p1和p2中哪个还有剩余的元素未排入help中,如果有则直接排入到help中。

void ExternalSort(int* arr, int L, int M, int R)
{
	int* help = (int*)malloc(sizeof(int) * (R - L + 1)); //辅助空间,用于存放排序后的数据,空间大小为R-L+1。
	if (help == NULL)
	{
		perror("ExternalSort->malloc");
		return;
	}
	int helpSz = R - L + 1;
	int i = 0;
	int p1 = L;
	int p2 = M + 1;
	while (p1 <= M && p2 <= R)
	{
        //判断p1是否小于等于p2,如果是则将p1指向的值放入help数组中然后两指针前进一位,反之p2亦然
		help[i++] = arr[p1] <= arr[p2] ? arr[p1++] : arr[p2++];
	}
	while (p1 <= M)   //如果p1还没越界,则将剩余的元素全部拷贝到help之后
	{
		help[i++] = arr[p1++];
	}
	while (p2 <= R)   //如果p2还没越界,则将剩余的元素全部拷贝到help之后
	{
		help[i++] = arr[p2++];
	}
	for ( i = 0; i < helpSz; i++)
	{
		arr[L + i] = help[i];    //将合并完成的数据拷贝回原数组arr的对应位置
	}
	free(help);
	help = NULL;
}

到这里,归并排序的代码实现部分就结束了,总的来说因为使用的是递归,代码量是不多的,但是最难的是理解归并排序的思路, 需要好好体会归并排序的操作步骤和思路。

3、master公式

那么完成了归并排序之后,我想知道这个排序的时间复杂度是多少的话,我该怎么算?有人说当然是直接百度搜索一下就知道了。我想说的是这样确实是没问题,但是秉持着“授人以鱼不如授人以渔”的理念,我想带大家深入了解并让大家学会自己去计算递归的时间复杂度

而用来计算的公式就是使用master公式:在计算涉及递归的算法的时候,计算复杂度就会变得有些麻烦。master公式就是用来进行剖析递归行为和递归行为时间复杂度的估算的。

3.1、公式以及结论

  • master公式:T(N) = a*T(N/b) + O(N^d)

  • T(N) = a * T(\frac{N}{b})+O(N^{d})

  • 公式解释:N表示母问题的规模。N/b表示子问题的规模,子问题规模必须相同,即都为N/b。a表示递归的次数也就是子问题在母问题中被调用了多少次。O(N^d)表示除了递归调用操作以外其余操作的复杂度。

  • 结论(证明过于复杂,只需要记住结论即可):
  1. 当公式中的a、b、d符合d<logb a时,时间复杂度为O(N^(logb a))
  2. 当公式中的a、b、d符合d=logb a时,时间复杂度为O((N^d)*logN)
  3. 当公式中的a、b、d符合d>logb a时,时间复杂度为O(N^d)
  • 注意:master公式适用于一些特殊的递归,就是子问题规模必须等分,不管你是分成几部分,就算是划分的区域有重叠,只要区域大小一致,就都可以使用。

【举例说明】

下面是一个使用递归实现在一个数组中找最大值的代码,这里是使用二分法来快速查找最大值,子问题的划分大小是一致的,因此可以使用master公式计算时间复杂度。

  1. 首先可以看到process中自身调用了两次process,母问题中有两个子问题调用即a=2
  2. 然后由于是二分法,因此每个子问题的规模是母问题规模的一半,即N/2
  3. 接着再看其他操作,其他操作都只执行一次,即时间复杂度为O(1)
  4. 最后:该递归的master公式就是T(N) = 2 * T(N/2) + O(1)。

即a = 2, b = 2, d = 0,代入结论公式得d<logb a,时间复杂度为O(N^(logb a)) = O(N)

int process(int* arr, int L, int R)
{
	if (L == R)
		return arr[L];
	int mid = L + (R - L) / 2;
	int leftMAX = process(arr, L, mid);    //左半边
	int rightMAX = process(arr, mid + 1, R);   //右半边
	return leftMAX > rightMAX ? leftMAX : rightMAX;
}

int main()
{
	int arr[] = { 8,3,6,4,2,1,5,7 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	int max = process(arr, 0, sz - 1);
	printf("%d\n", max);
	return 0;
}

3.2、适用于某些特殊的递归

上面说到过:master公式适用于一些特殊的递归,就是子问题规模必须等分,不管你是分成几部分,就算是划分的区域有重叠,只要区域大小一致,就都可以使用。

但是仍然会有些人会误解意思,下面使用图解的方式给大家解释一下。

【图解说明】

 首先,等分区域是最容易理解的,就是将N等分成若干份,二分就是N/2,三分就是N/3。

子问题的规模是左侧三分之二和右侧三分之二,这样的符合master公式吗?

答案是符合的,因为这里只关注的是区域大小是否一致,而不关心区域是否重叠。

 子问题规模不一样,不符合master公式。

3.3、计算归并排序的时间复杂度

学完了master公式,那么我们就来计算一下归并排序的时间复杂度吧。

void MergeSort(int* arr, int L, int R)
{
	if (L == R)  //子序列只有一个数,默认为有序
	{
		return;
	}
	int mid = L + (R - L) / 2;
	MergeSort(arr, L, mid);
	MergeSort(arr, mid + 1, R);
	ExternalSort(arr, L, mid, R);
}

假设整个过程的数据量是N的规模,两个子问题都是T(N/2),因此是2*T(N/2)。

那么现在来观察除了子问题外的其他语句:if语句是O(1)的时间复杂度,而ExternalSort函数的两个指针都只往右前进不会回退的遍历所有数据一遍,又因为数据量是N的规模,所以ExternalSort函数的时间复杂度是O(N)

即:T(N) = 2 * T(N/2) + O(N)    其中a  = 2,b = 2,d = 1

将a、b、d代入结论公式得d=logb a,时间复杂度为O((N^d)*logN) = O(N*logN)。

 

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!

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

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

相关文章

基于SpringBoot的桂林旅游景点导游平台

目录 前言 一、技术栈 二、系统功能介绍 用户信息管理 景点类型管理 景点信息管理 线路推荐管理 用户注册 线路推荐 论坛交流 三、核心代码 1、登录模块 2、文件上传模块 3、代码封装 前言 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实…

Unity基于种子与地块概率的开放世界2D地图生成

public class BuildingGen : MonoBehaviour {public int[] Building;//存储要生成的地块代码public int[] Probability;//存储概率public double seed;public int width 100;public int height 100;public float noiseScale 0.1f; //噪声缩放倍数private int[,] frequencyM…

基于Java的驾校收支管理可视化平台设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09;有保障的售后福利 代码参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作…

揭秘亚马逊、ebay自养号测评底层环境防关联技术

今天珑哥讲的是纯人工的自养号方式&#xff0c;已经在做过的人应该都懂&#xff0c;实现自养号所使用的IP和浏览器都有哪些 &#xff1f;都会有哪些问题。 一&#xff1a;市面上的IP有哪些&#xff1f;会遇到的问题&#xff1f; 922、luminati、googelfi、TM流量卡、Rola&…

基于Matlab求解高教社杯数学建模竞赛(cumcm2010A题)-储油罐的变位识别与罐容表标定(附上源码+数据+题目)

文章目录 题目解题源码数据下载 题目 通常加油站都有若干个储存燃油的地下储油罐&#xff0c;并且一般都有与之配套的“油位计量管理系统”&#xff0c;采用流量计和油位计来测量进/出油量与罐内油位高度等数据&#xff0c;通过预先标定的罐容表&#xff08;即罐内油位高度与储…

CentOS 7 停止维护后如何平替你的生产系统?

Author&#xff1a;rab 目录 前言一、Debian 家族1.1 Debian1.2 Ubuntu 二、RHEL 家族2.1 Red Hat Enterprise Linux2.2 Fedora2.3 CentOS2.4 Rocky Linux2.5 AlmaLinux 三、如何选择&#xff1f;思考&#xff1f; 前言 CentOS 8 系统 2021 年 12 月 31 日已停止维护服务&…

NPDP产品经理知识(产品创新种的市场调研)

1. 复习产品设计与开发工具 创意生成&#xff1a; scamper也叫蹦蹦法 心智图就是思维导图&#xff1a;mindmaping 原型法--故事板&#xff1a;创意生成的时候做的 人种学--民族志 六顶思考帽&#xff1a;白色红色黑色蓝色。。。 概念设计&#xff1a; AOMI&#xff1a;卡…

BIRCH算法全解析:从原理到实战

目录 一、引言什么是BIRCH算法BIRCH算法的应用场景文章目标和结构概述 二、BIRCH算法基础CF&#xff08;Clustering Feature&#xff09;树的概念数据点簇簇的合并和分裂 BIRCH的时间复杂度和空间复杂度BIRCH vs K-means和其他聚类算法 三、BIRCH算法的技术细节CF树的构建节点和…

潮流来袭!中国首届虚拟艺术巡展NFS将在广州YCC!天宜盛大开启!

中国首届虚拟艺术巡展 NFT Showcase (NFS) 即将来袭&#xff0c;本活动由 Web3 营销公司 Beep Crypto 精心策划&#xff0c;将于 10 月 5 日至 10 月 17 日在广州潮流策展空间 YCC!天宜举行。 活动的主题围绕虚拟时尚展开&#xff0c;汇聚了一系列令人激动的展品&#xff0c;涵…

【Java】抽象类

目录 概述&#xff1a; 示例代码&#xff1a; 抽象类特点 示例代码&#xff1a; 概述&#xff1a; 在Java中&#xff0c;一个没有 方法体 的方法应该定义为 抽象方法&#xff0c;而类中如果有 抽象方法&#xff0c;该类必须定义为抽象类。 抽象类和抽象方法用 abstract 关键…

基于Java的志愿者活动宣传管理系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09;有保障的售后福利 代码参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作…

再来介绍另一个binlog文件解析的第三方工具my2sql

看腻了文字就来听听视频演示吧&#xff1a;https://www.bilibili.com/video/BV1rp4y1w74B/ github项目&#xff1a;https://github.com/liuhr/my2sql gitee链接&#xff1a;https://gitee.com/mirrors/my2sql my2sql go版MySQL binlog解析工具&#xff0c;通过解析MySQL bin…

vertx的学习总结7

这里我就简单的聊几句&#xff0c;如何用vertx web来搞一个web项目的 1、首先先引入几个依赖&#xff0c;这里我就用maven了&#xff0c;这个是kotlinvertx web <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apac…

Linux基本指令介绍系列第四篇

文章目录 前言一、Linux基本指令介绍1、more指令2、less指令3、head指令4、tail指令5、bc指令6、管道文件介绍7、与时间相关的指令 总结 前言 本文介绍Linux使用时的部分指令&#xff0c;读者如果想了解更多基本指令的使用&#xff0c;可以关注博主的后续的文章。 博主使用的实…

基于Java的在线听书网站设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09;有保障的售后福利 代码参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作…

阿里云服务器搭建网站(图文新手教程)

使用阿里云服务器快速搭建网站教程&#xff0c;先为云服务器安装宝塔面板&#xff0c;然后在宝塔面板上新建站点&#xff0c;阿里云服务器网以搭建WordPress网站博客为例&#xff0c;来详细说下从阿里云服务器CPU内存配置选择、Web环境、域名解析到网站上线全流程&#xff1a; …

【Zookeeper专题】Zookeeper特性与节点数据类型详解

目录 前言前置知识课程内容一、Zookeeper介绍二、Zookeeper快速开始2.1 Zookeeper安装2.2 客户端命令行操作2.3 GUI工具 三、Zookeeper数据结构3.1 ZNode节点分类3.2 ZNode状态信息3.3 监听机制详解3.3.1 永久性Watch 3.4 节点ZNode特性总结3.5 应用场景详解3.5.1 统一命名服务…

大模型部署手记(7)LLaMA2+Jetson AGX Orin

1.简介 组织机构&#xff1a;Meta&#xff08;Facebook&#xff09; 代码仓&#xff1a;GitHub - facebookresearch/llama: Inference code for LLaMA models 模型&#xff1a;llama-2-7b、llama-2-7b-chat 下载&#xff1a;使用download.sh下载 硬件环境&#xff1a;Jetso…

字符函数和字符串函数(下)

目录 strncpy(Copy characters from string)函数的使用strncat(Append characters from string)函数的使用strncmp(Compare characters of two strings)函数的使用strstr(Locate substring)的使用和模拟实现 感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接 &am…

【HUAWEI】VLAN+OSPF+单臂路由

目录 &#x1f96e;写在前面 &#x1f96e;3.1、拓扑图 &#x1f96e;3.2、操作思路 &#x1f96e;3.3、配置操作 &#x1f363;3.3.1、LSW2配置 &#x1f363;3.3.2、LSW3配置 &#x1f363;3.3.3、R1配置 &#x1f363;3.3.4、R2配置 &#x1f363;3.3.5、LSW1配置 &#x1f…