归并排序-八大排序三大查找汇总(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 }
时间: 2024-12-28 02:12:57

归并排序-八大排序三大查找汇总(7)的相关文章

堆排序(选择排序)-八大排序三大查找汇总(2)

二叉堆的定义 二叉堆是完全二叉树或者是近似完全二叉树. 二叉堆满足二个特性: 1.父结点的键值总是大于或等于(小于或等于)任何一个子节点的键值. 2.每个结点的左子树和右子树都是一个二叉堆(都是最大堆或最小堆). 当父结点的键值总是大于或等于任何一个子节点的键值时为最大堆.当父结点的键值总是小于或等于任何一个子节点的键值时为最小堆.下图展示一个最小堆: 堆的存储 一般都用数组来表示堆,i结点的父结点下标就为(i – 1) / 2.它的左右子结点下标分别为2 * i + 1和2 * i + 2.如

简单选择排序(选择排序)-八大排序三大查找汇总(1)

工作原理: 每一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,直到全部待排序的数据元素排完. 稳定性: 选择排序是不稳定的排序方法(比如序列[5, 5, 3]第一次就将第一个[5]与[3]交换,导致第一个5挪动到第二个5后面). 时间复杂度: 比较次数O(n^2),比较次数与关键字的初始状态无关,总的比较次数N=(n-1)+(n-2)+...+1=n*(n-1)/2. 交换次数O(n),最好情况是,已经有序,交换0次:最坏情况下,即待排序记录初始状态是按第一条记录最大

快速排序(交换排序)-八大排序三大查找汇总(6)

基本思想 1.先从数列中取出一个数作为基准数. 2.分区过程,将比这个数大的数全放到它的右边,小于或等于它的数全放到它的左边. 3.再对左右区间重复第二步,直到各区间只有一个数. 性能 时间复杂度:平均情况下的时间复杂度为O(nlogn).最坏情况下时间复杂度为O(n2). 空间复杂度:除去程序运行实现的空间消费(例如递归栈),快速排序算法只需消耗确定数量的空间(即O(1),常数级空间). 稳定性:不稳定的算法 注意 编译器函数库自带的快速排序函数:qsort() 用 法: void qsort

冒泡排序-八大排序三大查找汇总(3)

基本思想 两两相邻元素之间的比较,如果前者大于后者,则交换: 设数组长度为N. 1.比较相邻的前后二个数据,如果前面数据大于后面的数据,就将二个数据交换. 2.这样对数组的第0个数据到N-1个数据进行一次遍历后,最大的一个数据就“沉”到数组第N-1个位置. 3.N=N-1,如果N不为0就重复前面二步,否则排序完成. 稳定性 冒泡排序是一种稳定的排序算法 时间复杂度 若文件的初始状态是正序的,一趟扫描即可完成排序.所需的关键字比较次数 和记录移动次数  均达到最小值:  ,  .所以,冒泡排序最好

希尔排序(插入排序)-八大排序三大查找汇总(5)

基本思想 该方法的基本思想是:先将整个待排元素序列分割成若干个子序列(由相隔某个“增量”的元素组成的)分别进行直接插入排序,然后依次缩减增量再进行排序,待整个序列中的元素基本有序(增量足够小)时,再对全体元素进行一次直接插入排序. 稳定性 由于多次插入排序,我们知道一次插入排序是稳定的,不会改变相同元素的相对顺序,但在不同的插入排序过程中,相同的元素可能在各自的插入排序中移动,最后其稳定性就会被打乱,所以shell排序是不稳定的. 时间复杂度 希尔排序的时间复杂度取决于步长的选择. 平均情况下,

直接插入排序(插入排序)-八大排序三大查找汇总(4)

基本思想 直接插入排序(Insertion Sort)的基本思想是:每次将一个待排序的记录,按其关键字大小插入到前面已经排好序的子序列中的适当位置,直到全部记录插入完成为止. 直接插入排序是由两层嵌套循环组成的.外层循环标识并决定待比较的数值.内层循环为待比较数值确定其最终位置.直接插入排序是将待比较的数值与它的前一个数值进行比较,所以外层循环是从第二个数值开始的.当前一数值比待比较数值大的情况下继续循环比较,直到找到比待比较数值小的并将待比较数值置入其后一位置,结束该次循环. 时间复杂度 O(

八大排序之归并排序

一.基本思想 归并排序,将当前序列分成若干个小的有序序列,然后逐个合并成更大的有序序列.这里所谓的若干个小的有序序列即是将序列分割成n个长度为1的序列,然后两两合并成长度为二的有序序列.然后在将这长度为二的有序序列合并为长度为四的有序序列.依次类推,最终达到原序列长度,这样,排序就完成了.这其实是归并排序递归回溯过程的描述.归并排序的正序描述可以是这样,将序列分成两个子序列,对两个子序列进行归并排序,然后将两个子序列合并到原序列中. 二.实现步骤 归并排序,有两个需要解决的问题:一.如何划分子区

数据结构常见的八大排序算法(详细整理)

https://www.jianshu.com/p/7d037c332a9d?utm_campaign=hugo&utm_medium=reader_share&utm_content=note&utm_source=weixin-friends 八大排序,三大查找是<数据结构>当中非常基础的知识点,在这里为了复习顺带总结了一下常见的八种排序算法.常见的八大排序算法,他们之间关系如下: 排序算法.png 他们的性能比较: 性能比较.png 下面,利用Python分别将他

八大排序算法原理以及Java实现(直接插入排序)

概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的排序记录,在排序过程中需要访问外存. 我们这里说说八大排序就是内部排序. 当n较大,则应采用时间复杂度为O(nlog2n)的排序方法:快速排序.堆排序或归并排序序. 快速排序:是目前基于比较的内部排序中被认为是最好的方法,当待排序的关键字是随机分布时,快速排序的平均时间最短: 1.插入排序-直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到