C语言手撕归并——递归与非递归实现(附动画及源码)

news2025/1/12 1:42:24

🤖💻👨‍💻👩‍💻🌟🚀

🤖🌟 欢迎降临张有志的未来科技实验室🤖🌟

专栏:数据结构 

👨‍💻👩‍💻 先赞后看,已成习惯👨‍💻👩‍💻

👨‍💻👩‍💻 创作不易,多多支持👨‍💻👩‍💻

🚀 启动创新引擎,揭秘C语言的密码🚀


🧠目录

归并排序概述

核心步骤

动画演示

归并排序的性能

递归代码实现

函数签名

逻辑流程

函数签名

逻辑流程

        非递归代码实现

msort函数解析


归并排序概述


归并排序的基本思想是将待排序的序列分成若干个子序列,每个子序列都是有序的。然后再合并这些子序列形成最终的排序序列。这个过程可以递归地进行,直到所有的子序列都只包含一个元素为止。

核心步骤

  1. 分解:将序列分为两半。

  2. 解决:递归地对每一半进行排序。

  3. 合并:将两个已排序的子序列合并成一个单一的有序的序列

动画演示

归并排序的性能

  • 最佳情况: 𝑂(𝑛log⁡𝑛)O(nlogn)
  • 平均情况: 𝑂(𝑛log⁡𝑛)O(nlogn)
  • 最差情况: 𝑂(𝑛log⁡𝑛)O(nlogn)
  • 空间复杂度: 𝑂(𝑛)O(n)
  • 稳定性: 归并排序是稳定的排序算法,即相等的元素在排序前后保持原有的相对顺序。

递归代码实现


函数签名

  • void merge(int arr[], int tempArr[], int left, int right, int mid)

int arr[]: 待排序的数组。

int tempArr[]: 一个临时数组,用于存储合并过程中产生的有序子数组。

int left: 左侧子数组的起始索引。

int right: 右侧子数组的结束索引。

int mid: 中间索引,用于分割数组。

逻辑流程
  • 初始化指针

int l_pos = left;

  • 初始化左侧子数组的指针 l_pos

int r_pos = mid + 1;

  • 初始化右侧子数组的指针 r_pos

int pos = left;

  • 初始化临时数组 tempArr 的指针 pos
  • 合并左右子数组

while (l_pos <= mid && r_pos <= right)

在 l_pos 和 r_pos 都没有超过各自子数组边界的情况下,比较左侧和右侧的元素。

如果左侧元素小于右侧元素,则将左侧元素放入 tempArr 中,并移动左侧指针;

否则,将右侧元素放入 tempArr 中,并移动右侧指针。

  • 处理左侧剩余元素

while (l_pos <= mid)

如果左侧子数组还有未处理的元素,则将这些元素依次放入 tempArr 中。

  • 处理右侧剩余元素

while (r_pos <= right)

如果右侧子数组还有未处理的元素,则将这些元素依次放入 tempArr 中。

  • 复制结果回原数组

for (int i = left; i <= right; i++)

将 tempArr 中的有序结果复制回原数组 arr 中对应的位置。

void merge(int arr[], int tempArr[], int left, int right, int mid)
{
	int l_pos = left;
	int r_pos = mid+1;
	int pos = left;

	while (l_pos <= mid && r_pos <= right){
		if (arr[l_pos] < arr[r_pos])
			tempArr[pos++] = arr[l_pos++];
		else
			tempArr[pos++] = arr[r_pos++];
	}

	while (l_pos <= mid) {
		tempArr[pos++] = arr[l_pos++];
	}

	while (r_pos <= right)
	{
		tempArr[pos++] = arr[r_pos++];
	}

	for (int i = left; i <= right; i++)
	{
		arr[i] = tempArr[i];
	}
}

函数签名

void msort(int arr[], int tempArr[], int left, int right)

  • int right: 数组的结束索引。
  • int left: 数组的起始索引。
  • int tempArr[]: 一个临时数组,用于存储中间结果,以便在合并阶段使用。
  • int arr[]: 待排序的数组。
逻辑流程
  • 递归终止条件

     if (right > left) 

  • 分割数组

 int mid = (left + right) / 2;

计算中间位置 mid,用来分割数组。这里使用 (left + right) / 2 来找到中间点。

  • 递归调用

msort(arr, tempArr, left, mid);

  • 对左边的子数组进行递归排序。

msort(arr, tempArr, mid + 1, right);

  • 对右边的子数组进行递归排序。
  • 合并排序后的子数组

merge(arr, tempArr, left, right, mid);

  • 调用 merge 函数来合并已经排序好的左右两个子数组。这个函数将两个已排序的子数组 arr[left...mid] 和 arr[mid+1...right] 合并成一个有序的数组。
void msort(int arr[], int tempArr[], int left, int right)
{
	if (right > left)
	{
		int mid = (left + right) / 2;
		msort(arr, tempArr, left, mid);
		msort(arr, tempArr, mid + 1, right);
		merge(arr, tempArr, left, right, mid);
	}
}

非递归代码实现

merge函数与上述类似,这里不再赘述

msort函数解析

  • 初始化步长

    • int gap = 1;
      • 初始化步长 gap,表示每次分割的子数组长度。
  • 外层循环

    • while (gap < right)
      • 在步长大于整个数组长度之前,继续进行循环。
  • 内层循环

    • for (int begin = 0; begin < right - gap; begin += 2 * gap)
      • 从数组的起始位置开始,每次增加 2 * gap,来遍历整个数组。
      • 这样做是为了确保每次合并的子数组不会重叠。
  • 确定子数组边界

    • int mid = begin + gap - 1;
      • 确定左侧子数组的结束位置 mid
    • int end = (begin + 2 * gap - 1) < right ? begin + 2 * gap - 1 : right;
      • 确定右侧子数组的结束位置 end。如果 begin + 2 * gap - 1 小于 right,则取 begin + 2 * gap - 1否则取 right
  • 调用 merge 函数

    • merge(arr, tempArr, begin, mid, end);
      • 调用 merge 函数来合并已排序的子数组。
  • 更新步长

    • gap *= 2;
      • 每次循环结束后,将步长翻倍,这样可以逐渐增加合并的子数组的大小。
    void msort(int* arr, int* tempArr, int left, int right)
    {
    	int gap = 1;
    	while (gap < right)
    	{
    		for (int begin = 0; begin < right - gap; begin += 2 * gap)
    		{
    			int mid = begin + gap - 1;
    			int end = (begin + 2 * gap - 1) < right ? begin + 2 * gap - 1 : right;
    			merge(arr, tempArr, begin, mid, end);
    		}
    		gap *= 2;
    	}
    }

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

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

相关文章

【Python基础】Python函数

本文收录于 《Python编程入门》专栏&#xff0c;从零基础开始&#xff0c;分享一些Python编程基础知识&#xff0c;欢迎关注&#xff0c;谢谢&#xff01; 文章目录 一、前言二、函数的定义与调用三、函数参数3.1 位置参数3.2 默认参数3.3 可变数量参数&#xff08;或不定长参数…

springboot项目--后端问题记录

springboot项目后端记录 前言一、包1. lombok--自动生成勾子方法作用依赖使用 2. Validated--自动校验作用依赖使用一般参数校验实体参数校验 结论 3. JWT(json web taken) 令牌生成什么是takenJWT包依赖使用获取taken校验 封装的工具类使用 二、处理技巧1. 全局异常处理作用代…

服务器禁用远程(22)

vim /etc/ssh/sshd_config 修改 ListenAddress 0.0.0.0 为ListenAddress localhost 修改完后 重启一下sshd systemctl restart sshd 修改就生效了

【 html+css 绚丽Loading 】000044 两仪穿行轮

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享htmlcss 绚丽Loading&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495…

数据库的配置2:客户端navicat的安装与连接数据库的方法

二.客户端的配置&#xff1a; 1.navicat ①安装&#xff1a; 破解根据文档进行破解即可 链接: https://pan.baidu.com/s/1M5KUv_fgRlFA50VfcV1VqA 提取码: j4a2 复制这段内容后打开百度网盘手机App&#xff0c;操作更方便哦 安装步骤很简单&#xff0c;直接下一步就好&…

Pandas处理数据,基本应用

Pandas是一个Python包&#xff0c;提供快速、灵活且表达力强的数据结构&#xff0c;旨在使处理“关系型”或“带标签”数据。专门设计用于进行数据分析和操作&#xff0c;它是建立在numpy之上&#xff0c;提供了易于使用的数据结构和数据分析工具。Pandas最主要的数据结构是Dat…

开启Hyper-V之后用不了VMware了,怎么破?

正文共&#xff1a;800 字 7 图&#xff0c;预估阅读时间&#xff1a;1 分钟 前面我们介绍了如何在Windows 10操作系统中启用Hyper-V虚拟化服务&#xff08;什么&#xff1f;Windows自带的Hyper-V虚拟化你都没用过&#xff1f;&#xff09;&#xff0c;但是在启用Hyper-V服务之…

vulhub spring 远程命令执行漏洞(CVE-2016-4977)

1.执行以下命令启动靶场环境并在浏览器访问 cd /vulhub/spring/CVE-2016-4977#进入漏洞环境所在目录 docker-compose up -d #启动靶场 docker ps #查看容器信息 2.输入以下命令测试环境 192.168.0.107:8080/oauth/authorize?response_type${2*2}&client_idacme&sc…

RMSE 和 RMS 介绍

RMSE&#xff08;Root Mean Square Error&#xff09;和 RMS&#xff08;Root Mean Square&#xff09;都是衡量误差或数据变动的统计量。它们在数据分析、机器学习和统计中应用广泛。以下是它们的详细介绍&#xff1a; 1. RMSE&#xff08;均方根误差&#xff09; 定义&…

【verilog】1. 流水灯例程

文章目录 前言一、定义概念 缩写1. verilog 二、性质三、代码分解释四、完整代码参考文献 前言 数电课设 一、定义概念 缩写 1. verilog Verilog 是一种以代码形式来描述数字系统和电路的硬件描述语言 (HDL)。它由 Gateway Design Automation 在 20 世纪 80年代中期开发&a…

9.06.

#include "mywidget.h"mywidget::mywidget(QWidget *parent): QMainWindow(parent) {/*---------------------窗口设置&#xff08;无边框&#xff09;----------------------*/this->setWindowFlag(Qt::FramelessWindowHint);//窗口大小this->resize(590,950)…

大数据之Flink(二)

4、部署模式 flink部署模式&#xff1a; 会话模式&#xff08;Session Mode&#xff09;单作业模式&#xff08;Per-Job Mode&#xff09;应用模式&#xff08;Application Mode&#xff09; 区别在于集群的生命周期以及资源的分配方式&#xff1b;以及应用的main方法到底在…

WireShark过滤器

文章目录 一、WireShark过滤器概念1. 捕获过滤器&#xff08;Capture Filters&#xff09;2. 显示过滤器&#xff08;Display Filters&#xff09;3. 捕获过滤器与显示过滤器的区别4. 过滤器语法结构实际应用场景 二、WireShark捕获数据包列表1. **No.&#xff08;序号&#xf…

vulhub ThinkPHP5 5.0.23远程代码执行漏洞

步骤一&#xff1a;.执行以下命令启动靶场环境并在浏览器访问 cd thinkphp/5.0.23-rcedocker-compose up -ddocker ps 步骤二&#xff1a;访问靶机环境 步骤三&#xff1a;/index.php?scaptcha 步骤四&#xff1a;利用HackBar _method__construct&filter[]system&me…

心理辅导新篇章:Spring Boot学生评估系统

1 绪论 1.1 研究背景 现在大家正处于互联网加的时代&#xff0c;这个时代它就是一个信息内容无比丰富&#xff0c;信息处理与管理变得越加高效的网络化的时代&#xff0c;这个时代让大家的生活不仅变得更加地便利化&#xff0c;也让时间变得更加地宝贵化&#xff0c;因为每天的…

优化边缘设备上的大型语言模型(LLM)--tinychat

文章目录 一、项目启动1.背景&#xff1a;针对不同操作系统架构的4bit权重重排2.初始环境配置下载LLaMA2-7B-chat模型 3.项目启动项目结构说明评估不同优化技术可能遇到的bug以及措施1.macOS上部署 二、各种优化技术实现1.前置条件2.优化----循环展开3.优化----多线程4.优化---…

OpenCV结构分析与形状描述符(6)带统计的连通组件计算函数connectedComponentsWithStats()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 connectedComponentsWithStats 函数计算布尔图像的连通组件标记图像&#xff0c;并为每个标记产生统计信息。 该函数接受一个具有4或8连通性的二…

盘点4款可以免费帮你将语音转换成文字的工具

我们在寻找语音转文字的工具的时候&#xff0c;不能只考虑他是否免费&#xff0c;还需要关注这个工具的转换准确度&#xff0c;减少第二次修改的麻烦&#xff0c;以及它的转换速度&#xff0c;以便可以有效的提高我们工作效率。基于这些&#xff0c;我要给大家推荐几个既可以免…

2024Java基础总结+【Java数据结构】(2)

面向对象07&#xff1a;简单小结类与对象 面向对象08&#xff1a;封装详解 面向对象09&#xff1a;什么是继承 ctrlh看类的关系&#xff0c;所有的类都默认的或间接继承Object 面向对象10&#xff1a;Super详解 super注意点: super调用父类的构造方法&#xff0c;必须在构造方…

白小白为波司登新品创作歌曲《登峰之路》,穿越风雨守护前行者

随着天气渐凉&#xff0c;波司登品牌推出全新新品——轻薄羽绒叠变系列&#xff0c;作为波司登品牌的新品推荐官&#xff0c;歌手白小白为波司登创作并演唱《轻薄羽绒叠变》系列主题曲《登峰之路》。歌曲中&#xff0c;白小白以激昂澎湃&#xff0c;明快有力的旋律以及深情又充…