Java算法之归并排序举例详解

 更新时间:2025年04月17日 09:38:15   作者:Brookty  
这篇文章主要介绍了Java算法之归并排序的相关资料,归并排序是一种递归排序算法,通过将数组分成更小的子数组,递归地排序这些子数组,然后将它们合并成有序数组,文中通过代码介绍的非常详细,需要的朋友可以参考下

一、归并排序的递归探寻

1.思路

理想结果等于成分解断开子结果表达式的公式表示

快速排序:

整个数组有序 = 其中一个元素有序 + 其左断开数组有序 + 其右断开数组有序

归并排序

整个数组有序 = 左断开数组有序 + 右断开数组有序 + 两有序数组的有序合并

2.搭建

2.1设计过掉不符情况(在最底层时)

  • if(array == null) return, 数组没有元素不用排,下面是有元素
  • if(left == right) return,只有一个元素已经是有序了不用排,下面是多个元素
  • if(left > right) return,排不了不要排的,之后下面是符合一般情况的多个元素 

2.2查验能实现基础排序(在最底层往上点时)

在最底层往上点时,有序数组有序合并操作在最底层能实现两元素之间的比较然后进行排序的

2.3跳转结果继续往上回搭:

跳转有序数组结果继续往上有序合并维护回搭

3.实质

从底层的最小单个断开有序数组往上有序地合并成越来越大的断开有序数组直至合并完成一个整体的有序数组

4.实现

    public static void mergeSort(int[] array) {
        mergeSortFunc(array,0,array.length-1);
    }

    private static void mergeSortFunc(int[] array,int left,int right) {
        if(left >= right) return;
        int mid = (left+right) / 2;

        mergeSortFunc(array,left,mid);
        mergeSortFunc(array,mid+1,right);
        merge(array,left,right,mid);
    }

    private static void merge(int[] array, int left, int right, int mid) {
        int s1 = left;
        int s2 = mid+1;
        int[] tmpArr = new int[right-left+1];
        int k = 0;
        //证明两个区间 都同时有数据的
        while (s1 <= mid && s2 <= right) {
            if(array[s2] <= array[s1]) {
                tmpArr[k++] = array[s2++];
            }else {
                tmpArr[k++] = array[s1++];
            }
        }
        while (s1 <= mid) {
            tmpArr[k++] = array[s1++];
        }
        while (s2 <= right) {
            tmpArr[k++] = array[s2++];
        }
        //tmpArr 里面一定是这个区间内有序的数据了
        for (int i = 0; i < tmpArr.length; i++) {
            array[i+left] = tmpArr[i];
        }
    }

二、递归的调用栈

1.递归的执行过程

在函数递归中,调用的函数里面执行着再调用着随着形参深入的不断变化的函数自己

第一次调用函数的执行转去等着第二次调用函数的执行完,第二次调用函数的执行也卡着转去等第三次调用函数的执行完,一层层形参变化着重复地执行调用而都没有往下去return直到最后调用函数传的形参变化到符合return的条件不再继续往下调用了return出结果开始往回地一层层促进上一层没有执行完到执行完return,即开始往回地执行往回地归

2.递归的函数栈帧

所有任意一个函数的调用都会独立开辟新的函数栈帧(里面存放局部变量、函数形参、返回地址、寄存器值)压入调用栈中,在函数执行到return语句结束后才弹出栈

递归调用时,函数栈帧从先往后地一个个独立地压入栈中,往下递归直到形参条件变到return由最新函数调用的栈帧开始往上弹栈帧出调用栈,在开始往上弹出栈帧开始有执行完往回层往下执行时,方法里面有可能写的是继续又去执行调用当前层形参条件下的再来一波函数递归调用(形参变化可能就去设置成不同的了就会是左右不一样的分支),即是二叉即二叉树的情况:

  • 每一层不仅要左边往下全执行到底层然后开始往上全执行到它那层的左边结束,接着又要执行右边的往下执行到底层然后再往上全执行完到它的右层结束,最后它这个节点对应的这层函数才也执行完它的栈帧也弹出调用栈,二叉树从大到小所有的结构都是左边往下全执行完往上回来、右边往下全执行完往上回来、接着它这个节点的栈帧也往上弹返回执行完 

2.1递归函数的栈帧压弹

在归并排序的二叉树递归调用过程中:

  • 每次累计着往下调用到底层时,此时的调用栈所占的空间是最大的、深度在二叉树的最底层,调用栈的空间计算为调用栈里栈帧的个数×每个栈帧的内存大小,在每个函数的栈帧中,函数里面那些函数的调用信息、并非循环出现的常量个数的局部变量空间和都可算成常数的大小,所以在归并排序这里,调用栈的最大空间为在调用栈里栈帧个数最多的时候:树的高度log(n)*每个函数栈帧的内存大小是常数,即log(n)*常数函数调用执行完最底层后就开始有往上返回了,往上弹出最新最顶的栈帧
  • 然后执行完返回到上层时又回到当前层条件下的且新的形参变化模式的再往下递归,又会去压栈到最底层,此时调用栈的空间又达到最大的log(n)
  • 当它右边往下的也全执行完又往上返回到当前层时,就开始继续往下接着执行就开始有去调用合并有序数组的函数了

2.2合并有序数组函数的栈帧压弹

执行调用合并有序数列的函数时,调用栈又会压入合并有序数组函数的栈帧,里面存放有开辟的当前层数组元素个数大小的数组(非常量级的,要算的),此时总的占用空间为调用栈的空间log(n-...)+n(-...),因为合并有序数组函数的栈帧每次都是处在栈顶压入的且函数里面并没有再调用函数的在它之上再压栈,所以它每次在栈顶进来压栈完就紧接着弹出栈的

三、归并排序的复杂度

1.空间复杂度

空间复杂度计算的是整个执行所有时刻中出现的最大瞬时占用空间

从下层往上层的返回的过程中,递归函数的调用栈空间变小着、合并有序数组的函数栈帧在变大着(里面的数组越来越大的):

  • 最底层时占用的空间为递归函数的调用栈空间log(n)+合并有序数组的函数栈帧0,即log(n)+0=log(n)
  • 当到达最上层第一层时,递归函数的调用栈空间是1,而合并有序数组的函数栈帧空间是n,此时的总空间大小是n,相比于最底层的log(n)及从下往上的过程中log(n)的递减、n的递增的总空间,此时的n是整个执行所有时刻中出现的最大瞬时占用的空间,所以归并排序的空间复杂度是O(n)

2.时间复杂度

时间复杂度即算整个递归调用执行过程的时间和,我们可以不用按着递归搜索的过程去时时累计总的算,直接站在总二叉树的角度一层一层地算所有时间的和就行了一层层里面每一个树节点及下的全执行完对应着该调用函数的全执行完,因为递归调用语句mergeSortFunc(array,left,mid)都是且已转成里面的函数节点内容来算了(调用中的去执行调用部分是常量级的已不算),且if(left >= right) return、int mid = (left+right) / 2也都是常量级的执行时间不算,对应到总的时间就是计算所有函数节点里的merge(array,left,right,mid)合并有序数组的时间和每一层所有函数节点的合并有序数组时间和都为n(除了最后一层的函数节点进去就直接判断为return没执行有序数组合并),一共有log(n)层,所以时间复杂度为O(n*log(n)) 

总结

到此这篇关于Java算法之归并排序的文章就介绍到这了,更多相关Java算法归并排序内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java线程重复执行以及操作共享变量的代码示例

    Java线程重复执行以及操作共享变量的代码示例

    这篇文章主要介绍了Java中对线程重复执行以及操作共享变量的代码示例,来自于Java面试题目的练习整理,需要的朋友可以参考下
    2015-12-12
  • 建议你使用LocalDateTime而不是Date哦

    建议你使用LocalDateTime而不是Date哦

    这篇文章主要介绍了建议你使用LocalDateTime而不是Date,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-10-10
  • java实现学生宿舍系统

    java实现学生宿舍系统

    这篇文章主要为大家详细介绍了java实现学生宿舍系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • java导出Excel文件的步骤全纪录

    java导出Excel文件的步骤全纪录

    这篇文章主要给大家介绍了关于java导出Excel文件的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用java具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-09-09
  • java爱心代码完整示例(脱单必备)

    java爱心代码完整示例(脱单必备)

    最近看到个好玩的,就是用代码实现爱心的形状,对于不懂编程的人来说,这是一个很好的玩的东西,这篇文章主要给大家介绍了关于java爱心代码的相关资料,需要的朋友可以参考下
    2023-07-07
  • Spring Boot 3.x 全新的热部署配置方式详解(IntelliJ IDEA 2023.1)

    Spring Boot 3.x 全新的热部署配置方式详解(IntelliJ ID

    这篇文章主要介绍了Spring Boot 3.x 全新的热部署配置方式(IntelliJ IDEA 2023.1),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-07-07
  • 详解如何在Java中创建Excel迷你图

    详解如何在Java中创建Excel迷你图

    迷你图是一种简洁而有效的数据可视化方式,常用于展示趋势和变化,通常被用于数据仪表盘、报告和展示中,以便在有限的空间内展示多个数据集的趋势,今天小编为大家介绍如何在Java中创建Excel迷你图,需要的朋友可以参考下
    2023-10-10
  • java web个人通讯录系统设计

    java web个人通讯录系统设计

    这篇文章主要为大家详细介绍了java web个人通讯录系统设计,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-01-01
  • Java 高并发三:Java内存模型和线程安全详解

    Java 高并发三:Java内存模型和线程安全详解

    本文主要介绍Java高并发内存模型和线程安全的资料,这里整理详细的资料及1.原子性 2.有序性 3.可见性 4.Happen-Before 5.线程安全的概念,有需要的小伙伴可以参考下
    2016-09-09
  • MyBatis中关于resultType和resultMap的区别介绍

    MyBatis中关于resultType和resultMap的区别介绍

    MyBatis中在查询进行select映射的时候,返回类型可以用resultType,也可以用resultMap,那么MyBatis中关于resultType和resultMap的区别是什么呢?下面小编通过本文给大家解答下
    2016-09-09

最新评论