C++归并排序详解以及代码实现

1. 介绍

归并排序(Merge Sort)是一种采用分治法(Divide and Conquer)策略的排序算法。该算法首先将已有序的子序列合并,得到完全有序的序列。在归并排序中,合并操作是将两个有序表合并成一个有序表的过程。

归并排序的原理是将数组不断分成两半,直到每个子数组只有一个元素,然后将这些子数组合并成一个有序的数组。合并操作需要两个子数组都是有序的,因此归并排序需要先对数组进行分解,然后再进行合并。

以下是归并排序的实现步骤:

  1. 将待排序的数组不断分成两半,直到每个子数组只有一个元素。这一步被称为分解。
  2. 对每个子数进行排序,可以使用任何有效的排序算法。这一步被称为求解。
  3. 将所有已排序的子数组合并成一个有序的数组。这一步被称为合并。

归并排序的时间复杂度是 O(n log n),其中 n 是数组的大小。这是因为它需要将数组分解成两半,然后再合并成一个有序的数组。归并排序的空间复杂度是 O(n),因为在合并过程中需要额外的空间来存储临时变量。

2. 代码实现

#include <iostream>  
using namespace std;  // 归并操作,将有序数组a和b合并成一个有序数组c  
void merge(int a[], int b[], int c[], int m, int n) {  int i = 0, j = 0, k = 0; // i、j、k分别指向数组a、b、c的起始位置  while (i < m && j < n) { // 比较a和b中的元素,将较小的元素放入c中  if (a[i] <= b[j]) {  c[k++] = a[i++];  } else {  c[k++] = b[j++];  }  }  while (i < m) { // 将数组a中剩余的元素放入c中  c[k++] = a[i++];  }  while (j < n) { // 将数组b中剩余的元素放入c中  c[k++] = b[j++];  }  
}  // 归并排序函数,将数组a中的元素进行排序  
void mergeSort(int a[], int n) {  if (n <= 1) { // 如果数组只有一个元素或为空,直接返回  return;  }  int mid = n / 2; // 计算数组的中间位置  int left[mid]; // 存储数组a左边部分的元素  int right[n - mid]; // 存储数组a右边部分的元素  for (int i = 0; i < mid; i++) { // 将数组a左边部分的元素放入left数组中  left[i] = a[i];  }  for (int i = mid; i < n; i++) { // 将数组a右边部分的元素放入right数组中  right[i - mid] = a[i];  }  mergeSort(left, mid); // 对left数组进行归并排序  mergeSort(right, n - mid); // 对right数组进行归并排序  merge(left, right, a, mid, n - mid); // 将left和right两个有序数组合并成一个有序数组a  
}  // 测试归并排序函数  
int main() {  int a[] = {38, 27, 43, 36, 16, 25, 6, 22}; // 待排序的数组  int n = sizeof(a) / sizeof(a[0]); // 数组的大小  mergeSort(a, n); // 对数组进行归并排序  for (int i = 0; i < n; i++) { // 输出排序后的数组元素  cout << a[i] << " ";  }  cout << endl;  return 0;  
}

上述代码实现了一个基于分治法的排序算法——归并排序,总结如下:

  1. 归并排序将数组不断分成两半,直到每个子数组只有一个元素。然后对每个子数组进行排序,最后将所有已排序的子数组合并成一个有序的数组。
  2. merge函数实现了合并操作,将两个有序的数组合并成一个有序的数组。
  3. mergeSort函数是归并排序的主要实现。首先判断数组长度,如果只有一个元素或为空,直接返回。然后计算中间位置,将数组分成左右两部分。接着对左右两部分递归地进行归并排序,最后调用merge函数将两个有序数组合并成一个有序数组。
  4. main函数中,定义了一个待排序的数组a,然后调用mergeSort函数对其进行归并排序。最后输出排序后的数组元素。
  5. 时间复杂度为O(n log n),空间复杂度为O(n)。

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

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

相关文章

echarts实现七天天气预报

效果图 实现代码 const imglist {"晴": data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAADMAAAAzCAYAAAA6oTAqAAAAAXNSR0IArs4c6QAAByJJREFUaEO9Wl1rFGcUfs67u/nYbOIXJlGopBTx27aQWGOwSMS75sao2BtBQVD/gWCJouA/sIWCBW8M1nqjd1IplW5DYqLWEUoSi9iEjHaJmaz2d05ZWd2Nj…

KingbaseV8R6单实例定时全量备份步骤

此场景为单机数据库节点内部备份&#xff0c;方便部署和操作&#xff0c;但备份REPO与数据库实例处于同一个物理主机&#xff0c;冗余度较低。 前期准备 配置ksql免密登录(必须) 在Kingbase数据库运行维护中&#xff0c;经常用到ksql工具登录数据库&#xff0c;本地免密登录…

基于OpenCV的图像颜色与形状识别的原理

基于 OpenCV 的图像颜色与形状识别是通过以下原理实现的&#xff1a; 图像预处理&#xff1a;首先&#xff0c;将彩色图像转换为灰度图像。这样做是因为在灰度图像中&#xff0c;每个像素只有一个颜色通道&#xff0c;可以更方便地进行后续处理。 阈值分割&#xff1a;对灰度图…

Linux系统编程(六):进程(下)

参考引用 UNIX 环境高级编程 (第3版)嵌入式Linux C应用编程-正点原子 1. 进程与程序 1.1 main() 函数由谁调用&#xff1f; C 语言程序总是从 main 函数开始执行int main(void) int main(int argc, char *argv[]) // 如果需要向应用程序传参&#xff0c;则选择该种写法操作系…

C++ 比 C语言增加的新特性 2

1.C新增了带默认值参数的函数 1.1 格式 格式&#xff1a;返回值 函数名&#xff08;参数1初始值1&#xff0c;..........&#xff09;{} 例如&#xff1a;void function&#xff08;int a10&#xff09;{} 调用&#xff1a;不需要更改参数的值&#xff1a;function&#x…

基于SSM和微信小程序的高校体育场管理系统

文章目录 项目介绍主要功能截图:部分代码展示设计总结项目获取方式🍅 作者主页:超级无敌暴龙战士塔塔开 🍅 简介:Java领域优质创作者🏆、 简历模板、学习资料、面试题库【关注我,都给你】 🍅文末获取源码联系🍅 项目介绍 基于SSM和微信小程序的高校体育场管理系…

文本编辑器:Sublime Text (安装+汉化)

下载 Sublime Text - Text Editing, Done Righthttps://www.sublimetext.com/Sublime Text官网 支持mac&#xff0c;Linux&#xff0c;Windows 安装 选择安装路径 next install 选择安装位置安装就行了 汉化 进入了主界面按 CTRLshiftp 输入install 选择第一个 弹窗就按确…

服务器扩容未生效、不成功:解决方法

记一次解决服务器扩容未生效的解决办法 老板&#xff1a;失忆啊&#xff0c;我花钱给服务器扩容了10000000G&#xff0c;但是数据库和mq都还是用不了&#xff0c;到底是不是服务器磁盘满了&#xff0c;你到底有没有查一下什么原因导致服务用不了啊。 失忆&#xff1a;老板您确…

概率论1:下象棋问题(3.5)

每日小语 时刻望着他人的眼色行事&#xff0c;是腾飞不了的。自己怎么想就积极地去做&#xff0c;这是需要胆量的。——广中平佑 题目 甲、乙二人下象棋&#xff0c; 每局甲胜的概率为a,乙胜的概率为b. 为简化问题&#xff0c;设没有和局的情况&#xff0c;这意味着a b1. 设想…

VR全景对普通人的生活有哪些好处?

许多普通人对VR全景还全然没有概念&#xff0c;这是因为VR全景虽然一直在快速发展&#xff0c;但目前为止也不过几年而已&#xff0c;但这发展的几年同样为我们普通人的生活带来了切实的改变和便利。VR全景技术为人们带来了沉浸感和真实感的体验&#xff0c;让我们感受到迥异于…

第十四章 集合(Set)

一、Set 接口&#xff08;P518&#xff09; 1. Set 接口基本介绍 &#xff08;1&#xff09;无序&#xff08;添加和取出的顺序不一致&#xff09;&#xff0c;没有索引。 &#xff08;2&#xff09;不允许重复元素&#xff0c;所以最多包含一个 null。 2. Set 接口的常用方法…

数据结构:KMP算法

1.何为KMP算法 KMP算法是由Knuth、Morris和Pratt三位学者发明的&#xff0c;所以取了三位学者名字的首字母&#xff0c;叫作KMP算法。 2.KMP的用处 KMP主要用于字符串匹配的问题&#xff0c;主要思想是当出现字符串不匹配时&#xff0c;我们可以知道一部分之前已经匹配过的的文…

【期刊周报1】医学好刊(SCI/SSCI/EI),含Top,领域广,接收快!

为了向广大学者朋友提供更优质的选刊服务&#xff0c;提高选刊质量&#xff0c;我处现开设周报专栏&#xff0c;以罗列我处合作的优质期刊~ 本期&#xff0c;小编给大家推荐的是医学领域相关的热门期刊&#xff0c;接收领域广&#xff0c;无预警&#xff0c;且在最新检索目录内…

Python遥感影像深度学习指南(2)-在 PyTorch 中创建自定义数据集和加载器

在上一篇 文章中,我们Fast.ai 在卫星图像中检测云轮廓,检测物体轮廓被称为语义分割。虽然我们用几行代码就能达到 96% 的准确率,但该模型无法考虑数据集中提供的所有输入通道(红、绿、蓝和近红外)。问题在于,深度学习框架(如 Keras、Fast.ai 甚至 PyTorch)中的大多数语…

油烟净化器如何做到高效净化?科技力量,清新餐饮生活

我最近分析了餐饮市场的油烟净化器等产品报告&#xff0c;解决了餐饮业厨房油腻的难题&#xff0c;更加方便了在餐饮业和商业场所有需求的小伙伴们。 油烟净化器的出现&#xff0c;为我们的餐饮生活注入了一抹清新的色彩。然而&#xff0c;它究竟是如何工作的&#xff1f;为何能…

【开题报告】基于SSM的健康饮食系统设计与实现

1.研究背景 如今&#xff0c;随着人们生活水平的提高和健康意识的增强&#xff0c;越来越多的人开始关注自己的饮食习惯&#xff0c;并希望通过合理的饮食来维持身体健康。然而&#xff0c;对于许多人来说&#xff0c;了解和选择合适的饮食方式并不容易。传统的饮食指导往往比…

【并发设计模式】聊聊Immutability模式利用不变性解决并发问题

上一篇文章&#xff0c;我们介绍了如何利用二阶段停止协议进行优雅停止线程和线程池&#xff0c;本篇介绍在并发编程中数据安全性&#xff0c;我们知道针对于数据的操作&#xff0c;读和写(添加、删除、修改), 在并发线程读写的时候&#xff0c;变量不加锁的情况下&#xff0c;…

redis哨兵+redis主从复制(在虚拟机centos的docker下)

1.安装docker Docker安装(CentOS)简单使用-CSDN博客 2.redis主从复制 redis主从复制(在虚拟机centos的docker下)-CSDN博客 3.编辑3个redis配置 cd /etc mkdir redis-sentinel cd redis-sentinel/ wget http://download.redis.io/redis-stable/sentinel.confcp sentinel.co…

ssh 免密登陆公钥设置失败分析调试

前景 看到这里肯定已经知道如何设置免密登陆。本文主要用于解决免密登陆设置失效问题。 ssh调试 目的 ssh设置了公钥仍然无法免密登陆; 需要调试 解决 通过systemctl status sshd的日志输出查看原因 步骤 打开调试 systemctl status sshd查看所在服务文件 $ sudo sys…

【并发编程篇】读锁readLock()和写锁writeLock()

文章目录 &#x1f6f8;情景引入⭐解决问题 readLock()和writeLock()都是ReadWriteLock接口中定义的方法&#xff0c;用于获取读锁和写锁。 readLock()方法返回一个读锁&#xff0c;允许多个线程同时获取该锁&#xff0c;以进行并发读取操作。如果当前已有一个写锁或其他线程正…