c++ 深入理解归并排序的用法

目录
  • 分治算法
  • 归并排序
  • 怎么分
  • 递归的出口
  • “并”的实现
  • 加到“分”函数里
  • 完整代码

hello

昨天发了个堆排序,竟然上了热榜

所以,今天来发一下归并排序

上次的堆排序似乎好多人没看懂,其实这些还是比较基础滴

废话不多说,直接进入正题

分治算法

如果你要学归并排序,首先你要学一下分治

所谓分治,就是分开治理,把大问题化成小问题,逐个解决,再合到一起

这也就是归并排序的精髓

这种算法时间复杂度低,原理也比较简单

归并排序

首先来看这张图

图片中把一个数组分成了一个一个的元素,在合并的过程中排序

怎么分

分的方法其实很简单,一个递归就可以解决

如果你是初学者,可能没有完全把递归学透彻

简单说,递归就是在函数内部调用自己的函数

递归都要有一个出口,否则就会变成死循环

递归的出口

我们在函数参数上写1)一个数组(要被排序的数组)2)分的开始和结束(first和end)

如果first<end,那么我们可以继续递归,如果不满足条件,递归结束

还要定义一个中间,前面那行代码是分左边,也就是开始~中间,后面那行代码是分右边,也就是中间+1~末尾


void merge_sort(int array[],int first,int end)
{
	if(first < end){
		int center = (first + end)/2;     //得到中间数
		merge_sort(array,first,center);
		merge_sort(array,center+1,end);
	}
}

“并”的实现

按照上面的图片,我们每排一下序就给它并一下

具体代码实现

void merge(int array[],int first,int center,int end)
{

	int n1 = center - first + 1;
	int n2 = end - center;
	int L[n1+1];
	int R[n2+1];
	for(int i = 0; i < n1; i++ )
	{
		L[i] = array[first+i];     //得到前面一部分数组
	}
	//printArray(L,n1);
	for(int j = 0; j < n2; j++ )
	{
		R[j] = array[center+j+1]; //得到后面一部分数组
	}
	//printArray(R,n2);
	L[n1] = 1000;    //设置哨兵
	R[n2] = 1000;	 //设置哨兵
	//cout << "R[5] =" << R[4] << endl;
	int k1 = 0;
	int k2 = 0;
	for (int k = first; k <= end; ++k)    //把得到的两个数组进行排序合并
	{
		//cout << L[k1] <<endl;
		//cout << R[k2] <<endl;
		if(L[k1] <= R[k2])
		{
			//cout << L[k1] <<endl;
			array[k] = L[k1];
			//cout << array[k] << endl;
			//cout << "k1 =" << k1 << endl;
			k1 = k1 + 1;
		}else{
			//cout << R[k2] <<endl;
			array[k] = R[k2];
			//cout << array[k] << endl;
			//cout << "k2 =" << k2 << endl;
			k2 = k2 + 1;
		}
		//cout << array[k] <<endl;
	}
	//printArray(array,10);
}

加到“分”函数里

因为我们分完了要并,所以我们把“并”函数写进“分”函数里

void merge_sort(int array[],int first,int end)
{
	if(first < end){
		int center = (first + end)/2;     //得到中间数
		merge_sort(array,first,center);
		merge_sort(array,center+1,end);
		merge(array,first,center,end);
	}
}

完整代码

加上int main()就行

#include <iostream>
using namespace std;

/*
* 打印数组
*/
void printArray(int array[],int length)
{
	for (int i = 0; i < length; ++i)
	{
		cout << array[i] << endl;
	}
}

/*
* 一个数组从中间分成两个有序数组
* 把这两个有序数组合并成一个有序数组
*/
void merge(int array[],int first,int center,int end)
{

	int n1 = center - first + 1;
	int n2 = end - center;
	int L[n1+1];
	int R[n2+1];
	for(int i = 0; i < n1; i++ )
	{
		L[i] = array[first+i];     //得到前面一部分数组
	}
	//printArray(L,n1);
	for(int j = 0; j < n2; j++ )
	{
		R[j] = array[center+j+1]; //得到后面一部分数组
	}
	//printArray(R,n2);
	L[n1] = 1000;    //设置哨兵
	R[n2] = 1000;	 //设置哨兵
	//cout << "R[5] =" << R[4] << endl;
	int k1 = 0;
	int k2 = 0;
	for (int k = first; k <= end; ++k)    //把得到的两个数组进行排序合并
	{
		//cout << L[k1] <<endl;
		//cout << R[k2] <<endl;
		if(L[k1] <= R[k2])
		{
			//cout << L[k1] <<endl;
			array[k] = L[k1];
			//cout << array[k] << endl;
			//cout << "k1 =" << k1 << endl;
			k1 = k1 + 1;
		}else{
			//cout << R[k2] <<endl;
			array[k] = R[k2];
			//cout << array[k] << endl;
			//cout << "k2 =" << k2 << endl;
			k2 = k2 + 1;
		}
		//cout << array[k] <<endl;
	}
	//printArray(array,10);
}

/*
* 分治算法
* 把一个数组从中间分成分开
* 然后进行排序
*/
void merge_sort(int array[],int first,int end)
{
	if(first < end){
		int center = (first + end)/2;     //得到中间数
		merge_sort(array,first,center);
		merge_sort(array,center+1,end);
		merge(array,first,center,end);
	}
}

int main(int argc, char const *argv[])
{
	int array[10] = {0,6,1,2,3,7,8,9,4,5};
	//merge(array,0,4,9);
	merge_sort(array,0,9);
	printArray(array,10);
	//int center = (0 + 9)/2;
	//cout << "center" << center << endl;
	//cout << "hello";
	return 0;
}

到此这篇关于c++ 深入理解归并排序的用法的文章就介绍到这了,更多相关c++ 归并排序内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C++实现归并排序

    定义:归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用.将已有序的子序列合并,得到完全有序的序列:即先使每个子序列有序,再使子序列段间有序.若将两个有序表合并成一个有序表,称为二路归并. 简单的来说,归并排序主要分为三步,一是对数组的划分,二是对数组的排序,三是对数组的合并.划分的大小是可以随自己的想法而设置,但是一般都是以2为单位,这样最小的一组的排序就比较方便. 具体一个简单的例子: 设有数

  • c++归并排序详解

    说一说归并排序 归并排序:归并排序(英语:Merge sort,或mergesort),是创建在归并操作上的一种有效的排序算法,效率为O(n log n).1945年由约翰·冯·诺伊曼首次提出.该算法是采用分治法(Divide and Conquer)的一个非常典型的应用,且各层分治递归可以同时进行. 归并排序的核心思想是将两个有序的数列合并成一个大的有序的序列.通过递归,层层合并,即为归并. 如图,从下到上,每一步都需要将两个已经有序的子数组合并成一个大的有序数组,如下是实现合并的具体代码,请

  • C++实现归并排序(MergeSort)

    本文实例为大家分享了C++实现归并排序的具体代码,供大家参考,具体内容如下 一.思路:稳定排序 (1)划分:一直调用划分过程,直到子序列为空或只有一个元素为止,共需log2(n): (2)归并:将两个子序列从小到大合并为一个序列 二.实现程序: // 归并排序:(二路归并) // (1)递归分解数组: // (2)合并有序的序列 #include <iostream> using namespace std; // 合并两个有序的序列 template <typename T> v

  • C++编程归并排序算法实现示例

    归并算法开始首先要对一段要有序的数字进行排序 void merg_sort(int* a, int fbegin, int fend, int sbegin, int send, int* b) { int L = fbegin; int R = sbegin; int cursize = fbegin;//z这里不能重0开始 递归后面是按对应开始位置进行赋值的 while (L <= fend && R <= send) { if (a[L] > a[R]) { b[c

  • C++归并排序算法详解

    目录 一.算法简介 二.实现过程 总结 一.算法简介 归并排序算法的平均时间复杂度是O(nlogn),归并算法的实现就是通过分冶法,将一个待排序列分成一个个小的序列,然后对这些小的序列进行排序,然后进行合并,合并的时候也会进行排序,这样,从整体拆成小块,再从小块合成整体的一个过程. 二.实现过程 1)拆分待排序列 2)进行排序合并 给大家写了一个简单的过程以便大家理解. 这基本就是归并排序的实现原理了,那么代码是怎么实现的呢,下面给大家展示下代码实现. //时间复杂度是nlogn #includ

  • c++实现二路归并排序的示例代码

    二路归并排序 基本思想 二路归并排序就是将两个有序子表归并成一个有序表.首先我们得有一个算法用于归并:两个有序表放在同一数组的相邻位置上,arr[left]到arr[center-1]为第一个有序表,arr[center]到arr[right]是第二个有序表.每次从两端中取出一个进行比较,小的先放在一个temp数组,最后将比较剩下的直接放到temp中去,最后将temp又复制回arr.这是"治". 所谓"分",就是递归地将前半部分和后半部分的数据各自归并排序即可. 算

  • C++实现归并排序算法

    归并 归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用.将已有序的子序列合并,得到完全有序的序列:即先使每个子序列有序,再使子序列段间有序.若将两个有序表合并成一个有序表,称为二路归并. 算法描述 归并操作的工作原理如下: 1.申请空间,使其大小为两个已经排序序列之和,该空间用来存放合并后的序列 2.设定两个指针,最初位置分别为两个已经排序序列的起始位置 3.比较两个指针所指向的元素,选择相对小

  • C++ 归并排序(merge sort)案例详解

    核心思想:"分"与"合". 主体流程 先将一个序列分成很多个不能再分割的子序列,将各个子序列分别排序后再将子序列合并.其实就是重复两个步骤:[1]分[2]合并. 首先是第一个小问题,怎么分? 比如说一个序列:12 ,23,1,44,233,10,9,8.我们先分成两段:12 ,23,1,44 和 233,10,9,8, 发现还能再分成4段:12 ,23 和 1,44------233,10 和 9,8. 再分成8段:12--23--1--44 和233--10--9

  • c++ 深入理解归并排序的用法

    目录 分治算法 归并排序 怎么分 递归的出口 "并"的实现 加到"分"函数里 完整代码 hello 昨天发了个堆排序,竟然上了热榜 所以,今天来发一下归并排序 上次的堆排序似乎好多人没看懂,其实这些还是比较基础滴 废话不多说,直接进入正题 分治算法 如果你要学归并排序,首先你要学一下分治 所谓分治,就是分开治理,把大问题化成小问题,逐个解决,再合到一起 这也就是归并排序的精髓 这种算法时间复杂度低,原理也比较简单 归并排序 首先来看这张图 图片中把一个数组分成了一个

  • 深入理解逻辑表达式的用法 与或非的用法

    先说逻辑与(&&),它可以从三个层次进行理解 第一个层次最简单,就是简单的布尔值之间的逻辑与,就是左值和右值都是true时,返回true,两边都是false或者两边的值其中一边是fasle,就返回false:(AND操作): 第二个层次,(false,null,indefined,0,-0,NaN和""这些都是假值,其他所有的值包括对象都是真值),对这些"真值"和"假值"进行AND操作,返回一个"真值"或者&q

  • Vuex之理解Mutations的用法实例

    1.什么是mutations? 上一篇文章说的getters是为了初步获取和简单处理state里面的数据(这里的简单处理不能改变state里面的数据),Vue的视图是由数据驱动的,也就是说state里面的数据是动态变化的,那么怎么改变呢,切记在Vuex中store数据改变的唯一方法就是mutation! 通俗的理解mutations,里面装着一些改变数据方法的集合,这是Veux设计很重要的一点,就是把处理数据逻辑方法全部放在mutations里面,使得数据和视图分离. 2.怎么用mutation

  • Vuex之理解Getters的用法实例

    1.什么是getters 在介绍state中我们了解到,在Store仓库里,state就是用来存放数据,若是对数据进行处理输出,比如数据要过滤,一般我们可以写到computed中.但是如果很多组件都使用这个过滤后的数据,比如饼状图组件和曲线图组件,我们是否可以把这个数据抽提出来共享?这就是getters存在的意义.我们可以认为,[getters]是store的计算属性. 2.如何使用 定义:我们可以在store中定义getters,第一个参数是state const getters = {sty

  • Vuex之理解Store的用法

    1.什么是Store? 上一篇文章说了,Vuex就是提供一个仓库,Store仓库里面放了很多对象.其中state就是数据源存放地,对应于与一般Vue对象里面的data(后面讲到的actions和mutations对应于methods). 在使用Vuex的时候通常会创建Store实例new Vuex.store({state,getters,mutations,actions})有很多子模块的时候还会使用到modules. 总结,Store类就是存储数据和管理数据方法的仓库,实现方式是将数据和方法

  • Vuex之理解state的用法实例

    1.什么是state? 上一篇文章说了,Vuex就是提供一个仓库,仓库里面放了很多对象.其中state就是数据源存放地,对应于与一般Vue对象里面的data(后面讲到的actions和mutations对应于methods). 响应书存储:state里面存放的数据是响应式的,Vue组件从store中读取数据,若是store中的数据发生改变,依赖这个数据的组件也会发生更新.(这里"状态"="数据"),也就是是说数据和视图是同步的. 2.局部状态 获取:在Vue组件中获

  • 五分钟理解keep alive用法及原理

    目录 引言 Home.vue About.vue App.vue 结合 Router,缓存页面 router.js Home.vue App.vue keep-alive 原理 引言 keep-alive可以实现组件缓存,当组件切换时不会对当前组件进行卸载. 主要是有include.exclude.max三个属性: 前两个属性允许keep-alive有条件的进行缓存: max可以定义组件最大的缓存个数,如果超过了这个个数的话,在下一个新实例创建之前,就会将以缓存组件中最久没有被访问到的实例销毁掉

  • python 切片和range()用法说明

    理解切片基本用法: 首先需要明白,可迭代对象,按照正数索引(正序)是从0开始的,按照负数索引(逆序)是从-1开始的.>>> astring = 'Hello world'>>> astring[0:2]'He'>>> 可见,这种情况下,给切片操作一个起始位置,和一个终止位置,则显示从起始位置开始(包括起始位置)到终止位置(不包括终止位置)之间的内容: 在有负数索引的情况下,是类似的,只要确定终止位置的内容: >>> astring[0

  • 深入理解PHP变量的值类型和引用类型

    在PHP中,大部分变量类型,如字符串,整型,浮点,数组等都是值类型的,而类和对象是引用类型,在使用的时候,需要注意这一点. 看到网友在讨论PHP的&符号,要彻底理解它的用法,就有必要讨论一下变量的两种形式. PHP的变量在内存中是这样存储的,变量保存的并不直接是值的内容,而是地址.例如: $a = 1; 我们看起来,似乎变量$a直接存储了 1 这个值.而实际情况是,PHP解释器创建了变量$a,将值:1 存入内存中的某个地方,再将值的地址存到变量$a中. 需要取值时,先找到变量$a中的地址,再根据

  • iOS开发中关键字const/static/extern、UIKIT_EXTERN的区别和用法

    一.前言 对于刚入行的新手们这些关键字可能会经常搞混淆或不清楚它们的意思和用法吧,即使在网上看了区别,但是很久不用下次又不清楚了,而且即使清楚自己的代码恐怕也很少用起来吧.通过阅读别人优秀的代码总会发现一些常用的关键字,随着自己的编程经验的积累慢慢的明白的. 二.关键字const/static/extern/UIKIT_EXTERN的释义和用法 1.const 这个单词翻译成中文是"常量"的意思.在程序中我们知道"常量"的值是不能变的,固定的.所以const关键字的

随机推荐