排序算法总结(四)归并排序

访问www.tomcoding.com网站,学习Oracle内部数据结构,详细文档说明,下载Oracle的exp/imp,DUL,logminer,ASM工具的源代码,学习高技术含量的内容。

归并排序的核心就是合并两个有序数组,生成一个有序的新数组。在一个数组中如果数字都无序,那么把这个数组拆成两个或四个数组,拆成的小数组都不能保证有序,如果继续拆下去,如果数组有n个元素,那么就拆成n个数组,这时每个数组只包含一个元素,这个单元素数组就是有序的了,然后把这些单元素数组两两合并,如果n是双数,那么正好两个配对,完成合并,如果n是单数,那么最后一个元素之前的正好两个配对,最后一个元素与之前合并好有序数组再合并,这样全部元素都合并成了有序数组。

我们来看合并两个有序数组的算法。算法原理很简单,在这里使用一个辅助数组c,两个有序数组分别叫做a和b,都设置一个指针指向首元素,首先a和b的首个元素进行比较,小的元素放到c数组的第一个元素,比较中第一个元素小的数组指针前进一个元素,这个元素又成了首元素,这样一直比较下去,小的元素放到c数组,指针往前移一个元素,直到两个数组中有一个指针移到了尾部,这时把没有移到尾部的数组剩余的元素依次放入c数组,这样c数组中的元素就都是有序的了。

在实际的应用中,往往a和b两个数组是一个大数组前后两部分,这时a和b是连在一起的,比如大数组有8个元素1,3,5,7,2,4,6,8,拆分成a和b两个数组,a的元素为1,3,5,7,b的元素为2,4,6,8,其实a就是大数组的首地址。所以上面的c数组中的元素要拷贝回a数组中,便于后面更大的有序数组合并。

合并函数如下。

/* ai是要合并的两个数组首地址

* left是数组首元素下标,right是数组最后元素下标,mid是数组中间元素下标

* 两个数组中的元素分别是left到mid,mid+1到right,这两个数组元素都是有序的

* 比如ai={1, 3, 5, 7, 2, 4, 6, 8}, left=0, mid=3, mid+1=4,right=7

* left到mid的元素是1, 3, 5, 7

* mid+1到right的元素是2, 4, 6, 8

* tmp是辅助临时数组地址

*/

void merge(int *ai, int *tmp, int left, int mid, int right)

{

int i, j, k;

/* i是左边数组的指针,j是右边数组的指针,

* 从头开始比较,较小的元素放入辅助数组tmp中

* 任何一个数组到达尾部,退出循环

*/

for (i=left, j=mid+1, k=0; (i<=mid) && (j<=right); ) {

if (ai[i] < ai[j])

tmp[k++] = ai[i++];

else

tmp[k++] = ai[j++];

}

/* 左边的数组没比较完,把左边剩余的元素放入辅助数组 */

while (i <= mid)

tmp[k++] = ai[i++];

/* 右边的数组没比较完,把右边剩余的元素放入辅助数组 */

while (j <= right)

tmp[k++] = ai[j++];

/* 把辅助数组中的元素拷贝回原始数组中 */

for (i=0, j=left; j<=right; j++, i++)

ai[j] = tmp[i];

}

把数组中的每个元素拆分成单元素的数组,是在排序函数中完成的,使用递归的方式。函数如下。

/* ai是要排序的数组

* left是数组第一个元素的下标,right是数组最后一个元素的下标

* tmp是一个与ai大小相同的辅助数组

*/

void merge_sort(int *ai, int *tmp, int left, int right)

{

int mid;

/* 左边的下标值与右边的相等了,

* 表示数组中只有一个元素了,不要再拆分了

*/

if (left >= right)

return;

/* 得到中间值 */

mid = (left + right) / 2;

/* 一直拆分左边的数组,直到每个数组包含一个元素

* 这儿是递归调用,其实是每次到了这儿,又会调用自己把自己拆分成左右两个数组

* 直到最后是单元素数组时,函数返回,这时又把小数组逐渐合并成大数组

* 左半部分合并成有序的大数组后,才会继续下面的函数,又是拆分再合并的过程

*/

merge_sort(ai, tmp, left, mid);

/* 一直拆分右边的数组,直到每个数组包含一个元素 */

merge_sort(ai, tmp, mid+1, right);

/* 合并两个有序的数组,从小到大,直到合并成一个有序的大数组 */

merge(ai, tmp, left, mid, right);

}

相关推荐
int型码农3 小时前
数据结构第八章(一) 插入排序
c语言·数据结构·算法·排序算法·希尔排序
怀旧,3 小时前
【数据结构】6. 时间与空间复杂度
java·数据结构·算法
积极向上的向日葵3 小时前
有效的括号题解
数据结构·算法·
Java 技术轻分享4 小时前
《树数据结构解析:核心概念、类型特性、应用场景及选择策略》
数据结构·算法·二叉树··都差速
chao_7895 小时前
链表题解——两两交换链表中的节点【LeetCode】
数据结构·python·leetcode·链表
曦月逸霜6 小时前
第34次CCF-CSP认证真题解析(目标300分做法)
数据结构·c++·算法
开开心心就好7 小时前
高效Excel合并拆分软件
开发语言·javascript·c#·ocr·排序算法·excel·最小二乘法
吴声子夜歌9 小时前
OpenCV——Mat类及常用数据结构
数据结构·opencv·webpack
笑口常开xpr9 小时前
数 据 结 构 进 阶:哨 兵 位 的 头 结 点 如 何 简 化 链 表 操 作
数据结构·链表·哨兵位的头节点
XRZaaa10 小时前
常见排序算法详解与C语言实现
c语言·算法·排序算法