• 归并排序-八大排序汇总(7)


    基本思想

      归并排序简单的说就是递归后合并,该算法是分治法(Divide and Conquer)的一个典型应用。

      基本思想为:将待排序序列R[0...n-1]看成是n个长度为1的有序序列,两两有序表成对归并,得到n/2个长度为2的有序表;将这些有序序列再次归并,如此反复进行下去,最后得到一个长度为n的有序序列。

      综上可知:

      归并排序其实要做两件事:

      (1)“分解”——将序列每次折半划分

      (2)“合并”——将划分后的序列段两两合并后排序

    性能

    排序类别 排序方法 时间复杂度 空间复杂度 稳定性 复杂性
    平均情况 最坏情况 最好情况
    归并排序 归并排序 O(nlog2n) O(nlog2n) O(nlog2n) O(n) 稳定 较复杂

      

     
     
      时间复杂度:归并排序的形式就是一棵二叉树,它需要遍历的次数就是二叉树的深度,而根据完全二叉树的可以得出它的时间复杂度是O(n*log2n)。
      空间复杂度:为 O(n),算法处理过程中,需要一个大小为n的临时存储空间用以保存合并序列。
      稳定性:在归并排序中,相等的元素的顺序不会改变,所以它是稳定的算法。
    比较
      归并排序比较占用内存,但却是一种效率高且稳定的算法。和堆排序、快速排序的比较:

      若从空间复杂度来考虑:首选堆排序,其次是快速排序,最后是归并排序。

      若从稳定性来考虑,应选取归并排序,因为堆排序和快速排序都是不稳定的。

      若从平均情况下的排序速度考虑,应该选择快速排序。

    代码及分析

      首先考虑下如何将将二个有序数列合并。这个非常简单,只要从比较二个数列的第一个数,谁小就先取谁,取了后就在对应数列中删除这个数。然后再进行比较,如果有数列为空,那直接将另一个数列的数据依次取出即可。

    //将有序数组a[]和b[]合并到c[]中
    void MemeryArray(int a[], int n, int b[], int m, int c[])
    {
        int i, j, k;
    
        i = j = k = 0;
        while (i < n && j < m)
        {
            if (a[i] < b[j])
                c[k++] = a[i++];
            else
                c[k++] = b[j++]; 
        }
    
        while (i < n)
            c[k++] = a[i++];
    
        while (j < m)
            c[k++] = b[j++];
    }

      解决有序序列的合并以后进行递归调用即为归并。

     1 //将有二个有序数列a[first...mid]和a[mid...last]合并。
     2 void mergearray(int a[], int first, int mid, int last, int temp[])
     3 {
     4     int i = first, j = mid + 1;
     5     int m = mid,   n = last;
     6     int k = 0;
     7     
     8     while (i <= m && j <= n)
     9     {
    10         if (a[i] <= a[j])
    11             temp[k++] = a[i++];
    12         else
    13             temp[k++] = a[j++];
    14     }
    15     
    16     while (i <= m)
    17         temp[k++] = a[i++];
    18     
    19     while (j <= n)
    20         temp[k++] = a[j++];
    21     
    22     for (i = 0; i < k; i++)
    23         a[first + i] = temp[i];
    24 }
    25 void mergesort(int a[], int first, int last, int temp[])
    26 {
    27     if (first < last)
    28     {
    29         int mid = (first + last) / 2;
    30         mergesort(a, first, mid, temp);    //左边有序
    31         mergesort(a, mid + 1, last, temp); //右边有序
    32         mergearray(a, first, mid, last, temp); //再将二个有序数列合并
    33     }
    34 }
    35 
    36 bool MergeSort(int a[], int n)
    37 {
    38     int *p = new int[n];
    39     if (p == NULL)
    40         return false;
    41     mergesort(a, 0, n - 1, p);
    42     delete[] p;
    43     return true;
    44 }

     自己写的代码:

     1 void mergearray1(int a[], int first, int mid, int last)
     2 {
     3     int i = first, j = mid + 1, m = mid, n = last;
     4     int tmp[14],k=0;
     5 
     6     while (i <= m && j <= n)
     7     {
     8         if (a[i] <= a[j])
     9             tmp[k++] = a[i++];
    10         else
    11             tmp[k++] = a[j++];
    12     }
    13         while (i <= m)
    14             tmp[k++] = a[i++];
    15         while (j<=n)
    16             tmp[k++] = a[j++];
    17 
    18     for (int h = 0; h < k; h++)
    19     {
    20         a[first+h] = tmp[h];
    21     }
    22 }
    23 
    24 void merge1(int a[], int first, int last)
    25 {
    26     if (first < last)
    27     {
    28         int mid = (first + last) / 2;
    29         merge1(a,first,mid);
    30         merge1(a,mid+1,last);
    31         mergearray1(a,first,mid,last);
    32     }
    33 }
    34 
    35 int _tmain(int argc, _TCHAR* argv[])
    36 {
    37     int a[] = { 12, 34, 21, 55, 27, 32, 33, 53, 22, 14, 4, 2, 6, 9 };
    38     merge1(a,0,13);
    39 
    40     for (int m = 0; m < 14; m++)
    41     {
    42         cout << a[m] << ",";
    43     }
    44     cout << endl;
    45 
    46     system("pause");
    47     return 0;
    48 }
  • 相关阅读:
    C语言关键字register、extern、static、一些总结,及项目中使用的心得
    c语言,文件操作总结
    《Redis内存数据库》Redis内存数据库技术总结
    《Redis内存数据库》Redis环境搭建(Linux)
    《Linux 操作系统》Linux的常用命令操作大全
    《Java练习题》Java编程题合集(全)
    《Java练习题》Java习题集一
    《Java基础知识》Java技术总结
    《Java基础知识》Java数据类型以及变量的定义
    《Java 底层原理》Jvm 类的加载原理
  • 原文地址:https://www.cnblogs.com/SnailProgramer/p/4854922.html
Copyright © 2020-2023  润新知