归并排序

作者: Joe_HUST | 来源:发表于2017-09-04 13:26 被阅读0次

归并排序是利用归并的思想对数列进行排序。--将两个有序数列合并成一个有序数列,称之为“归并”,归并包括从上往下和从下往上两种方式。

  1. 从下往上的归并排序 :将待排序数组分成若干个长度为1的子数列,然后再将这些数列两两合并,得到若干个长度为2的有序数列,再将这些数列两两合并;得到若干个长度为4的有序数列,再将它们两两合并直到合并为一个数列为止,这样就得到最终结果。
  2. 从上往下的归并排序 :它与从上往下的在排序方向上是相反的,包括三步:
    (1): 分解----将当前区间一分为二,即求分裂点mid=(low+high)/2;
    (2): 求解----递归的对两个子区间a[low...mid]和a[mid+1...high]进行归并排序,递归的终结条件是子区间的长度为1.
    (3):合并----将已经排好序的两个子区间a[low...mid]和a[mid+1...high]归并为一个有序的区间a[low...high].

归并排序的时间复杂度和稳定性

归并排序的时间复杂度是O(NlgN):归并排序的形式是一颗二叉树,他需要遍历的次数就是二叉树的深度。而根据完全二叉树可以得出他的时间复杂度是O(NlgN).

归并排序的稳定性

归并排序是稳定的算法


1.从上往下的归并排序

void Merge(int *a,int start,int mid,int end,int *tmp)
{
    int i=start;
    int j=mid+1;
    int k =0;
    while(i<=mid && j<=end)
    {
        if(a[i]<=a[j])
            tmp[k++] = a[i++];
        else
            tmp[k++] = a[j++];
    }

    while(i<=mid)
        tmp[k++] = a[i++];

    while(j<=end)
        tmp[k++] = a[j++];  

    for (i = 0; i<k; ++i)
        a[start+i] = tmp[i];
    return ;
}
void Merge_Step(int *a,int start,int end,int *tmp)
{
    if(start<end)
    {
        // int mid = (end+start)/2;
        // int mid = start+(end-start)/2;
                int mid = start+((end-start)>>1);
        //这里和上面是一样的但是可以解决end和start都很大的情况下,start+end溢出的情况
        Merge_Sort(a,start,mid,tmp);
        Merge_Sort(a,mid+1,end,tmp);
        Merge(a,start,mid,end,tmp);
    }
}

void Merge_Sort(int *a,int len)
{
    int i=0;
    int n=len-1;
    int *tmp = (int *)malloc(len*sizeof(int));
    Merge_Step(a,0,n,tmp);
    free(tmp);
}

2.从下往上的归并排序

void Merge(int *a,int start,int mid,int end,int *tmp)
{
    int i=start;
    int j=mid+1;
    int k =0;
    while(i<=mid && j<=end)
    {
        if(a[i]<=a[j])
            tmp[k++] = a[i++];
        else
            tmp[k++] = a[j++];
    }
    while(i<=mid)
        tmp[k++] = a[i++];

    while(j<=end)
        tmp[k++] = a[j++];  

    for (i = 0; i<k; ++i)
        a[start+i] = tmp[i];
    return ;
}
void Merge_step(int *a,int len,int step,int *tmp)
{
    int i =0;
    for (; i+2*step-1 < len; i+=2*step)
        Merge(a,i,i+step-1,i+2*step-1,tmp);

    if (i+step-1 < len-1)
        Merge(a,i,i+step-1,len-1,tmp);
}

int Merge_Sort(int *a,int len)
{
    assert(a);
    int *tmp = (int *)malloc(len*sizeof(int));
    if(NULL == tmp)
        return -1;

    int i=1;
    for (; i < len; i<<1)
        Merge_step(a,len,i,tmp);
    free(tmp);
}

相关文章

  • 排序算法

    约定 选择排序 冒泡排序 插入排序 希尔排序 归并排序1. 归并方法2. 自顶向下归并排序3. 自底向上归并排序 ...

  • 排序二:归并、快排

    文章结构 归并排序 快速排序 源码 1. 归并排序 1.1 什么是归并排序 归并排序的思想是:将待排序的区间平分成...

  • java归并排序

    归并排序什么是归并排序:图解归并排序归并排序有两种实现方式,一是基于递归,而是基于迭代1)基于递归的归并排序: 基...

  • 算法—排序篇2

    1、归并排序(Merging Sort) 归并排序(Merging Sort): 就是利用归并的思想实现排序⽅法....

  • 常见的排序算法(2)

    要点 快速排序 归并排序 1.快速排序 2.归并排序

  • 排序算法之归并排序

    归并排序(Merge Sort) 归并排序是利用归并的思想实现排序的方式,该算法采用的是经典的分治算法 归并排序过...

  • 算法 第二章第二部分笔记

    各种排序算法的性能特点 选择排序 插入排序 希尔排序 归并排序 本地归并排序 自底向上的归并排序 快速排序 三向切...

  • 归并排序(二路归并排序)

    归并排序的思路 归并排序是通过“归并”操作完成排序的,将两个或者多个有序子表归并成一个子表。归并排序是“分治法”的...

  • 算法排序之归并排序和快速排序

    归并排序和快速排序用的都是分治的思想,用递归的编程技巧来实现.咱们先来看归并排序. 归并排序 归并排序的核心思想就...

  • 基于左闭右开的乱序数组归并排序 2020-04-24(未经允许,

    归并排序代码模板 递归形式思路:二分nums数组后对nums的归并排序 = 对左侧数组归并排序+对右侧数组归并排序...

网友评论

    本文标题:归并排序

    本文链接:https://www.haomeiwen.com/subject/huhjjxtx.html