【数据结构】归并排序

devtools/2024/9/24 11:14:36/

1、介绍

归并排序(merge sort)是一种基于分治策略的算法>排序算法,包含“划分”和“合并”阶段。

  1. 划分阶段:通过递归不断地将数组从中点处分开,将长数组的排序问题转换为短数组的排序问题。

  2. 合并阶段:当子数组长度为 1 时终止划分,开始合并,持续地将左右两个较短的有序数组合并为一个较长的有序数组,直至结束。

2、算法流程

“划分阶段”从顶至底递归地将数组从中点切分为两个子数组。

  1. 计算数组中点 mid ,递归划分左子数组(区间 [left, mid] )和右子数组(区间 [mid + 1, right] )。

  2. 递归执行步骤 1. ,直至子数组区间长度为 1 时终止。

“合并阶段”从底至顶地将左子数组和右子数组合并为一个有序数组。需要注意的是,从长度为 1 的子数组开始合并,合并阶段中的每个子数组都是有序的。

归并排序与二叉树后序遍历的递归顺序是一致的。

  • 后序遍历:先递归左子树,再递归右子树,最后处理根节点。

  • 归并排序:先递归左子数组,再递归右子数组,最后处理合并。

归并排序的实现如以下代码所示。请注意,nums 的待合并区间为 [left, right] ,而 tmp 的对应区间为 [0, right - left]

/*合并左子数组和右子数组 */
void merge(vector<int>& nums,int left, int mid, int right)
{// 左子数组区间为[left,mid],右子数组区间为[mid+1,right]// //创建一个临时数组tmp,用于存放合并后的结果vector<int> tmp(right - left + 1);//初始化左右子数组的起始索引int i = left, j = mid + 1, k = 0;//当左右子数组都还有元素时,进行比较并将较小的元素复制到临时数组中while (i <= mid && j <= right){if (nums[i] <= nums[j]){tmp[k++] = nums[i++];}else{tmp[k++] = nums[j++];}}//将左子数组和右子数组的剩余元素复制到临时数组中while (i <= mid){tmp[k++] = nums[i++];}while (j <= right){tmp[k++] = nums[j++];}//将临时数组tmp中的元素复制回原数组nums的对应区间for(k = 0;k < tmp.size();k++){nums[left + k] = tmp[k];}
}void mergeSort(vector<int>& nums, int left, int right)
{//终止条件if (left >= right){return;}//划分阶段int mid = left + (right - left) / 2;   //计算划分中点mergeSort(nums, left, mid);//递归左子数组mergeSort(nums, mid + 1, right);//递归右子数组//合并阶段merge(nums,left, mid, right);
}

3、算法特性

  • 时间复杂度为 O(nlog⁡n)、非自适应排序:划分产生高度为 log⁡n 的递归树,每层合并的总操作数量为 n ,因此总体时间复杂度为 O(nlog⁡n) 。

  • 空间复杂度为 O(n)、非原地排序:递归深度为 log⁡n ,使用 O(log⁡n) 大小的栈帧空间。合并操作需要借助辅助数组实现,使用 O(n) 大小的额外空间。

  • 稳定排序:在合并过程中,相等元素的次序保持不变。


http://www.ppmy.cn/devtools/96754.html

相关文章

1.Linux_常识

UNIX、Linux、GNU 1、UNIX UNIX是一个分时操作系统&#xff0c;特点是多用户、多任务 实时操作系统&#xff1a;来了请求就去解决请求 分时操作系统&#xff1a;来了请求先存着&#xff0c;通过调度轮到执行时执行 2、Linux Linux是一个操作系统内核 发行版本&#xff1…

C++ //练习 17.5 重写findBook,令其返回一个pair,包含一个索引和一个迭代器pair。

C Primer&#xff08;第5版&#xff09; 练习 17.5 练习 17.5 重写findBook&#xff0c;令其返回一个pair&#xff0c;包含一个索引和一个迭代器pair。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具&#xff1a;vim 代码块 vector<vector<Sal…

测试流程自动化实践!

测试流程自动化的最佳实践涉及多个方面&#xff0c;旨在提高测试效率、确保测试质量&#xff0c;并降低测试成本。以下是一些关键的实践方法&#xff1a; 1. 明确测试目标 确定测试范围&#xff1a;在开始自动化测试之前&#xff0c;需要明确哪些功能、模块或场景需要被测试。…

Keepalived 高可用集群详解和配置

Keepalived 高可用集群 集群类型 1、LB&#xff08;Load Balance&#xff09;&#xff1a;负载均衡 LVS&#xff1a;四层负载均衡 HAProxy&#xff1a;七层/四层 负载均衡 nginx&#xff1a;七层负载均衡 (http/upstream,stream/upstream) 2、HA&#xff08;High Availa bili…

C#高级应用

C# 特性&#xff08;Attribute&#xff09; 特性&#xff08;Attribute&#xff09;是用于在运行时传递程序中各种元素&#xff08;比如类、方法、结构、枚举、组件等&#xff09;的行为信息的声明性标签。您可以通过使用特性向程序添加声明性信息。一个声明性标签是通过放置在…

依赖倒置原则详解

依赖倒置原则详解 一、引言 在大型系统架构设计中&#xff0c;依赖倒置原则&#xff08;Dependency Inversion Principle&#xff0c;DIP&#xff09;被广泛视为增强系统灵活性和可维护性的核心原则之一。最近在架构设计审查中&#xff0c;我们经常遇到由于依赖关系设计不当导…

机器学习:逻辑回归原理、参数介绍和优缺点

1、概念 逻辑回归是一种统计方法&#xff0c;用于分析一个或多个自变量&#xff08;解释变量&#xff09;与一个二元因变量&#xff08;响应变量&#xff09;之间的关系。虽然称为“回归”&#xff0c;但逻辑回归实际上是一种分类算法&#xff0c;因为它的输出是类别标签&#…

Icecream Video Converter Pro v1.44 激活版下载安装教程 (专业的视频转换软件)

前言 Icecream Video Converter Pro是一款高性能的视频转换软件&#xff0c;支持多种视频格式的转换。其用户界面设计简洁易用&#xff0c;提供高质量的视频转换和编辑功能。借助GPU加速和多核处理技术&#xff0c;该软件大大提升了转换速度和效率&#xff0c;同时确保视频转换…