排序——4.归并排序

归并排序

  开始之前,看这样一个例子:假设桌子上有两堆扑克牌,每堆只有一张,我们比较这两张扑克牌,将小的那个放到一个新堆里,再把大的那个放到小的那个上边。其实这样就完成了一次简单的归并排序。

  先通过上边的例子了解下什么是归并排序,再往下看就容易理解了。

  刚刚的例子里将要合并的两个部分都只有一个元素,下面是对于多个多元素的合并:

  假设我们已经有了两个早已经排序好了的数组,分别是a[N],b[M]。数组a中的元素从下标0到下标N-1逐渐增大(a[0]最小,a[N-1]最大),数组b也是这样。

1.我们先将a[0]和b[0]比较,先把小的那个放到一个新数组c中(c[0]=a[0],假设a[0]<b[0])然后再把a[0]和b[0]中较大的那个放到新数组c中(c[1]=b[0])。

2.再比较a[1]和b[1],,先把小的那个放到一个新数组c中(c[2]=a[1],假设a[1]<b[1])然后再把a[0]和b[0]中较大的那个放到新数组c中(c[3]=b[1])。

3.以此类推直到所有元素都放到c中。这样就将两个已经排好序的数组合并到一起了,并且也排好序了。

有了上边这个方法,我们就可以实现归并排序了。归并排序总结为一句话就是:先分解,再合并。每次对数组一分为二,将分解开的部分再对半分,直到分解到每部分都只剩一个元素,然后再按照开始的那个例子逆着合并回去。

正式开始:

分解

假设有数组a[10]={8,4,6,3,2,1,8,5,11,25}。

先将它一分为二,拆解成这样:

再把得到的这两部分,分别一分为二从,拆解成这样:

一直这样分下去,直到每个部分只有一个元素:

(颜色差点不够用的···)

下图这样更直观一些(分解过程):

等到分解到每个部分都只有一个元素了,就要开始合并了,回想最开始的那个例子,逆着合并回去。

直接看图吧:

代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace 归并排序
{
    class Program
    {
        static void Main(string[] args)
        {
            int[] a = new int[] { 8, 4, 6, 3, 2, 1, 8, 5, 11, 25 };
            Console.WriteLine("未排序之前的顺序:");
            foreach (int s in a)
            { Console.WriteLine("    {0}", s); }

            int[] b = new int[10];
            MergeSort(a, b, 0, a.Length-1 );

            Console.WriteLine("排序之后的顺序:");
            foreach (int s in a)
            { Console.WriteLine("    {0}", s); }
            Console.ReadKey();
        }
        public static void MergeSort(int[] sourceArr, int[] tempArr, int startIndex, int endIndex)
        {
            int midIndex;
            if (startIndex < endIndex)
            {
                midIndex = (startIndex + endIndex) / 2;
                MergeSort(sourceArr, tempArr, startIndex, midIndex);
                MergeSort(sourceArr, tempArr, midIndex + 1, endIndex);
                Merge(sourceArr, tempArr, startIndex, midIndex, endIndex);
            }
        }
        public static void Merge(int[] sourceArr, int[] tempArr, int startIndex, int midIndex, int endIndex)
        {
            int L = startIndex, R = midIndex + 1, k = startIndex;
            while (L != midIndex + 1 && R != endIndex + 1)
            {
                if (sourceArr[L] >= sourceArr[R])
                    tempArr[k++] = sourceArr[R++];
                else
                    tempArr[k++] = sourceArr[L++];
            }
            while (L != midIndex + 1)//某一方已经结束,合并另一方剩下所有
                tempArr[k++] = sourceArr[L++];
            while (R != endIndex + 1)//某一方已经结束,合并另一方剩下所有
                tempArr[k++] = sourceArr[R++];
            for (int i = startIndex; i <= endIndex; i++)
                sourceArr[i] = tempArr[i];
        }
    }
}

就算明白了归并排序的原理,转化成代码的过程也没有想象中那么简单,做了一个小视频来演示以上代码在合并的时候是如何工作的,该视频还有一些瑕疵,并不能完全代表实际工作过程,仅用作参考。

http://v.youku.com/v_show/id_XMTQ4NDc5NTUzNg==.html

时间: 2024-11-06 09:26:34

排序——4.归并排序的相关文章

数据结构教程 第三十六课 选择排序、归并排序

教学目的: 掌握选择排序,归并排序算法 教学重点: 选择排序之堆排序,归并排序算法 教学难点: 堆排序算法 授课内容: 一.选择排序 每一趟在n-i+1(i=1,2,...n-1)个记录中选取关键字最小的记录作为有序序列中第i个记录. 二.简单选择排序 算法: Smp_Selecpass(ListType &r,int i) { k=i; for(j=i+1;j<n;i++) if (r[j].key<r[k].key) k=j; if (k!=i) { t=r[i];r[i]=r[k

内部排序:归并排序和快速排序

前言   之所以把归并排序和快速排序放在一起探讨,很明显两者有一些相似之处:这两种排序算法都采用了分治的思想.下面来逐个分析其实现思想. 归并排序 实现思想 归并的含义很明显就是将两个或者两个以上的有序表组合成一个新的有序表.归并排序中一般所用到的是2-路归并排序,即将含有n个元素的序列看成是n个有序的子序列,每个子序列的长度为1,而后两两合并,得到n/2个长度为2或1的有序子序列,再进行两两合并...直到最后由两个有序的子序列合并成为一个长度为n的有序序列.2-路归并的核心操作是将一维数组中前

算法速成(三)七大经典排序之直接插入排序、希尔排序和归并排序

直接插入排序: 这种排序其实蛮好理解的,很现实的例子就是俺们斗地主,当我们抓到一 手乱牌时,我们就要按照大小梳理扑克,30秒后, 扑克梳理完毕,4条3,5条s,哇塞......  回忆一下,俺们当时是怎么梳理的. 最左一张牌是3,第二张牌是5,第三张牌又是3, 赶紧插到第一张牌后面去,第四张牌又是3,大喜,赶紧插到第二张后面去, 第五张牌又是3, 狂喜,哈哈,一门炮就这样产生了. 怎么样,生活中处处都是算法,早已经融入我们的生活和 血液. 下面就上图说明: 看这张图不知道大家可否理解了,在插入排

Java排序算法 归并排序

归并排序(Merge)是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的.然后再把有序子序列合并为整体有序序列. 归并排序是建立在归并操作上的一种有效的排序算法.该算法是采用分治法(Divide and Conquer)的一个非常典型的应用.将已有序的子序列合并,得到完全有序的序列:即先使每个子序列有序,再使子序列段间有序.若将两个有序表合并成一个有序表,称为2-路归并. 归并排序算法稳定,数组需要O(n)的额外空间,链表需要O(log(n))的

Javascript排序算法之合并排序(归并排序)的2个例子_基础知识

归并排序(Merge sort)是建立在归并操作上的一种有效的排序算法.该算法是采用分治法(Divide and Conquer)的一个非常典型的应用. 归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的.然后再把有序子序列合并为整体有序序列. 归并排序是建立在归并操作上的一种有效的排序算法.该算法是采用分治法(Divide and Conquer)的一个非常典型的应用.将已有序的子序列合并,得到完全有序的序列:即先使每个

java实现归并排序和树形排序(锦标赛制):java字符串分隔或的形式

String[] b=str.split("query|,");//query分隔或者逗号分隔 归并排序,递归实现 public class MergeSort2 { // 对data数组中的 [a,b) 区间的数据进行归并排序, // 排序结束后,[a,b)间数据处于升序有序状态 static void mergeSort(int[] data, int a,int b) { if (a >= b) return; int mid=(a+b)/2;//拆分排序 mergeSor

内部排序算法:归并排序

基本思想 n个记录的文件的直接选择排序可经过n-1趟直接选择排序得到有序结果: 初始状态:无序区为R[1..n],有序区为空. 第1趟排序: 在无序区R[1..n]中选出关键字最小的记录R[k],将它与无序区的第1个记录R[1] 交换,使R[1..1]和R[2..n]分别变为记录个数增加1个的新有序区和记录个数减少1个的新无序区. -- 第i趟排序: 第i趟排序开始时,当前有序区和无序区分别为R[1..i-1]和R[i..n](1≤i≤n-1). 该趟排序从当前无序区中选出关键字最小的记录R[k

C语言实现排序算法之归并排序详解_C 语言

排序算法中的归并排序(Merge Sort)是利用"归并"技术来进行排序.归并是指将若干个已排序的子文件合并成一个有序的文件. 一.实现原理: 1.算法基本思路 设两个有序的子文件(相当于输入堆)放在同一向量中相邻的位置上:R[low..m],R[m+1..high],先将它们合并到一个局部的暂存向量R1(相当于输出堆)中,待合并完成后将R1复制回R[low..high]中. (1)合并过程 合并过程中,设置i,j和p三个指针,其初值分别指向这三个记录区的起始位置.合并时依次比较R[i

各种排序算法汇总

目录 简介 交换排序 冒泡排序 快速排序 插入排序 直接插入排序 希尔排序 选择排序 简单选择排序 堆排序 归并排序 基数排序 总结 简介 排序是计算机内经常进行的一种操作,其目的是将一组"无序"的记录序列调整为"有序"的记录序列.分内部排序和外部排序.若整个排序过程不需要访问外存便能完成,则称此类排序问题为内部排序.反之,若参加排序的记录数量很大,整个序列的排序过程不可能在内存中完成,则称此类排序问题为外部排序.内部排序的过程是一个逐步扩大记录的有序序列长度的过程