深入浅出排序算法之归并排序

目录

[1. 归并排序的原理](#1. 归并排序的原理)

[1.1 二路归并排序执行流程](#1.1 二路归并排序执行流程)

[2. 代码分析](#2. 代码分析)

[2.1 代码设计](#2.1 代码设计)

[3. 性能分析](#3. 性能分析)

[4. 非递归版本](#4. 非递归版本)


1. 归并排序的原理

"归并"一词的中文含义就是合并、并入的意思,而在数据结构中的定义是将两个或者两个以上的有序表组合成一个新的有序表。

**归并排序(Merging Sort)**就是利用归并的思想实现的排序方法。它的原理是假设初始序列含有n个记录,则可以看成是n个有序的子序列,每个子序列的长度为1,然后两两归并,得到[n/2](表示不小于x的最小整数)个长度为2或1的有序子序列;再两两归并,......,如此重复,直至得到一个长度为n的有序序列为止,这种排序方法成为2路归并排序。

1.1 二路归并排序执行流程

原始序列:49 38 65 97 76 13 27

(1)将原始序列看成7个只含有一个关键字的子序列,显然这些子序列都是有序的。

子序列1:49;子序列2:38;子序列3:65;子序列4:97;子序列5:76;子序列6:13;子序列7:27

(2)两两归并,形成若干有序二元组 ,即49和38归并成{38 49},65和97归并成{65 97},76和13归并成{13 76},27没有归并对象,保持原样{27}。第一趟二路归并排序结束,结果如下:

{38 49},{65 97},{13 76},{27}

(3)再将这个序列看成若干二元组子序列

子序列1:38 49;子序列2:65 97;子序列3:13 76;子序列4:27;

最后一个子序列长度可能是1,也可能是2。

(4)继续两两归并,形成若干有序四元组(同样,最后的子序列中不一定有4个关键字),即{38 49}和{65 97}归并形成{38 49 65 97},{13 76}和{27}归并形成{13 27 76}。第二趟二路归并排序结束,结果如下:

{38 49 65 97},{13 27 76}

(5)最后只有两个子序列了,再进行一次归并,就可完成整个二路归并排序,结果如下:

13 27 38 49 65 76 97

2. 代码分析

大家先看看有没有思路!

3,2,1 开始我的表演哈哈哈!

请看图解:

2.1 代码设计

根据图解我们要怎么设计方法呢?

**我打算把分解功能写成一个方法,合并功能写成一个方法。**具体实现如下:

java 复制代码
    public static void mergeSort1(int[] array){
        mergeSortDivide(array,0, array.length - 1);
    }

    //分解
    private static void mergeSortDivide(int[] array,int left,int right){
        //终止条件:left > right
        while(left >= right){
            return;
        }
        int mid = (left+right)/2;
        //递归左子序列
        mergeSortDivide(array,left,mid);
        //递归右子序列
        mergeSortDivide(array,mid+1,right);
        //合并
        merge(array,left,right,mid);
    }

    //合并
    private static void merge(int[] array,int start,int end,int mid){
        //左子序列从start开始
        int s1 = start;
        //右子序列从mid+1开始
        int s2 = mid + 1;
        //新建一个数组,作为复制数组
        int[] tmp = new int[end - start + 1];
        //k表示中间数组的元素下标
        int k = 0;
        //开始比较
        while(s1 <= mid && s2 <= end){
            if(array[s1] <= array[s2]){
                //将小的值赋值给tmp[k]
                //小伙伴们这可以思考一下:为什么不能先写array[s2] <= array[s1]?
                //等下看我下面的解析!
                tmp[k++] = array[s1++];
            } else{//array[s2] <= array[s1]的情况
                tmp[k++] = array[s2++];
            }
        }
        //有剩余的数组
        //左子序列
        while(s1 <= mid){
            tmp[k++] = array[s1++];
        }
        //右子序列
        while(s2 <= end){
            tmp[k++] = array[s2++];
        }
        //将tmp数组的值赋值给array数组
        for(int i = 0;i<tmp.length;i++){
            array[i+start] = tmp[i];
        }
    }

回答问题:为什么不能先写array[s2] <= array[s1]?

答:归并排序是稳定的。如果先写array[s2] <= array[s1],那么在s2开始的元素与s1开始的元素相等的话,例如:1<=1,那么本该在后面的1就会移到前面,导致这段代码实现的归并排序不稳定了!

3. 性能分析

|--------------|-----------|
| 时间复杂度 | 空间复杂度 |
| O(n*log(n)) | O(n) |

4. 非递归版本

上面的版本是递归版本,接下来是非递归版本。

java 复制代码
    private static void merge(int[] array,int start,int end,int mid) {
        int s1 = start;
        //int e1 = mid;
        int s2 = mid+1;
        //int e2 = end;
        int[] tmp = new int[end-start+1];
        int k = 0;//tmp数组的下标
        while (s1 <= mid && s2 <= end) {
            if(array[s1] <= array[s2]) {
                tmp[k++] = array[s1++];
            }else {
                tmp[k++] = array[s2++];
            }
        }
        while (s1 <= mid) {
            tmp[k++] = array[s1++];
        }
        while (s2 <= end) {
            tmp[k++] = array[s2++];
        }

        for (int i = 0; i < tmp.length; i++) {
            array[i+start] = tmp[i];
        }

    }

    public static void mergeSort(int[] array) {
        int gap = 1;
        while (gap < array.length) {
            // i += gap * 2 当前gap组的时候,去排序下一组
            for (int i = 0; i < array.length; i += gap * 2) {
                int left = i;
                int mid = left+gap-1;//有可能会越界
                if(mid >= array.length) {
                    mid = array.length-1;
                }
                int right = mid+gap;//有可能会越界
                if(right>= array.length) {
                    right = array.length-1;
                }
                merge(array,left,right,mid);
            }
            //当前为2组有序  下次变成4组有序
            gap *= 2;
        }
    }
相关推荐
京东零售技术3 小时前
扛起技术大梁的零售校招生们 | 1024技术人特别篇
算法
爱coding的橙子3 小时前
每日算法刷题Day78:10.23:leetcode 一般树7道题,用时1h30min
算法·leetcode·深度优先
Swift社区3 小时前
LeetCode 403 - 青蛙过河
算法·leetcode·职场和发展
地平线开发者3 小时前
三种 Badcase 精度验证方案详解与 hbm_infer 部署实录
算法·自动驾驶
papership3 小时前
【入门级-算法-5、数值处理算法:高精度的减法】
算法·1024程序员节
lingran__4 小时前
算法沉淀第十天(牛客2025秋季算法编程训练联赛2-基础组 和 奇怪的电梯)
c++·算法
DuHz4 小时前
基于MIMO FMCW雷达的二维角度分析多径抑制技术——论文阅读
论文阅读·物联网·算法·信息与通信·毫米波雷达
一二学长4 小时前
快速排序(JAVA详细讲解快速排序的四种方式)
数据结构
Dragon_D.4 小时前
排序算法大全——插入排序
算法·排序算法·c·学习方法
大数据张老师5 小时前
数据结构——红黑树
数据结构·算法·红黑树