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

访问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);

}

相关推荐
ZSYP-S1 小时前
Day 15:Spring 框架基础
java·开发语言·数据结构·后端·spring
唐叔在学习1 小时前
【唐叔学算法】第21天:超越比较-计数排序、桶排序与基数排序的Java实践及性能剖析
数据结构·算法·排序算法
ALISHENGYA2 小时前
全国青少年信息学奥林匹克竞赛(信奥赛)备考实战之分支结构(switch语句)
数据结构·算法
武昌库里写JAVA4 小时前
浅谈怎样系统的准备前端面试
数据结构·vue.js·spring boot·算法·课程设计
S-X-S5 小时前
代码随想录刷题-数组
数据结构·算法
l138494274515 小时前
每日一题(4)
java·数据结构·算法
kyrie_sakura5 小时前
c++数据结构算法复习基础--13--基数算法
数据结构·c++·算法
XWXnb65 小时前
数据结构:顺序表
数据结构·算法
橘颂TA5 小时前
【C++】数据结构 顺序表的实现(详解)
开发语言·数据结构·c++·算法
R_.L5 小时前
数据结构:双向循坏链表
数据结构·链表