归并排序图解

简介: 归并排序图解

七大排序之归并排序

前言

博主个人社区: 开发与算法学习社区

博主个人主页:Killing Vibe的博客

欢迎大家加入,一起交流学习~~

一、算法思路

归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide andConquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。

动图如下:
在这里插入图片描述

其实就是分为两个过程:

归: 不断将原数组拆分为子数组(一分为二),直到每个子数组只剩下一个元素 = 》 归过程结束

并:不断合并相邻的两个子数组为一个大的子数组,合并的过程就是将两个已经有序的子数组合并为一

举个栗子:

如下图所示的待排序序列,经过不断的归过程变成了一个个的小数组:
在这里插入图片描述

在这里插入图片描述

当每个子数组都只剩下一个元素时,每个子数组都是一个有序数组,然后开始两两合并。

如下是并过程

在这里插入图片描述

时间复杂度:稳定O(nlogn) 空间复杂度:O(n)

二、代码实现

代码如下:

    private static void mergeSortInternal(int[] arr, int l, int r) {
        // 2.小数组直接使用插入排序
        if (r - l <= 15) {
            insertionSort(arr,l,r);
            return;
        }
        // int mid = (l + r) / 2;
        // l = 2,r = 4  mid = 3 = 2 + (4 - 2) / 2 = 3
        int mid = l + ((r - l) / 2);
        mergeSortInternal(arr,l,mid);
        mergeSortInternal(arr,mid + 1,r);
        // arr[l..mid] 和 arr[mid + 1...r]有序 只需要合并这两个子数组即可
        // 1.到底什么时候才需要合并 arr[mid] < arr[mid + 1] 说明?arr[mid] 数组1的最大值 arr[mid + 1]数组2的最小值
        // 整个数组已经有序了,那你还合并个der
        if (arr[mid] > arr[mid + 1]) {
            // 前后两个子数组还存在乱序,才需要合并
            merge(arr,l,mid,r);
        }
    }


    /**
     * 将有序子数组arr[l..mid] 和 有序arr[mid + 1...r] 合并为一个大的有序数组arr[l..r]
     * @param arr
     * @param l
     * @param mid
     * @param r
     */
    private static void merge(int[] arr, int l, int mid, int r) {
        // 先创建一个新的数组aux,将子数组的值复制给新数组
        int[] aux = new int[r - l + 1];
        // l = 2,r = 4
        // arr[2..4]
        // aux[0..2] 索引下标差了个l偏移量
        for (int i = 0; i < aux.length; i++) {
            // aux的索引下标0...arr.length - 1
            // arr的下标l...r
            aux[i] = arr[i + l];
        }
        // 数组1的开始下标
        int i = l;
        // 数组2的开始下标
        int j = mid + 1;
        for (int k = l; k <= r; k++) {
            if (i > mid) {
                // 第一个数组已经遍历完毕
                arr[k] = aux[j - l];
                j ++;
            }else if (j > r) {
                // 第二个子数组遍历完毕
                arr[k] = aux[i - l];
                i ++;
            }else if (aux[i - l] <= aux[j - l]) {
                // 将aux[i - l]写回arr[k]
                arr[k] = aux[i - l];
                i ++;
            }else {
                // aux[i - l] > aux[j - l] 写回aux[j - l]
                arr[k] = aux[j - l];
                j ++;
            }
        }
    }


    /**
     * 在arr[l..r]上进行插入排序
     * @param arr
     * @param l
     * @param r
     */
    private static void insertionSort(int[] arr, int l, int r) {
        for (int i = l + 1; i <= r; i++) {
            for (int j = i; j >= l + 1 && arr[j] < arr[j - 1]; j--) {
                swap(arr,j,j - 1);
            }
        }
    }

注意:这是优化后的归并排序,小数组直接使用插入排序能更加优化时间。

因为插入排序在近乎有序的数组时间复杂度很低接近O(n)。

没优化之前的代码可以把此处代码更换掉:

在这里插入图片描述
改为:

if(r <= l) return;

三、算法稳定性

最核心的merge操作: 需要开辟额外空间,空间大小就是合并后的数组大小

  1. 先将两个子数组的所有内容复制到新数组中
  2. 遍历两个子数组,将较小值写回原数组

两边都从头开始遍历,将较小值写回arr数组即可,如下:

在这里插入图片描述

四、拓展

4.1 海量数据处理:用到外部存储器

内存只有1G,待排序的数据有100G,该如何对这100G的数据进行排序?

a. 先把这个100G的数据分成200份

b. 分别将0.5G的数据读入内存,进行内部排序(归并,快排,堆排)

c. 进行200个文件的merge操作即可

整个结果就有序了~

4.2 归并排序的衍生问题

归并排序非常适合对链式结构进行排序,链表,二叉树等~

这里有两道💪扣的题目,可以用归并排序分治的思想去思考,下面推给大家:

  • 剑指offer 51 : 数组中的逆序对 (困难)
  • 力扣148 :排序链表 (中等)

总结

以上就是归并排序的图解和代码,有什么疑问可以私信博主~有帮助的话可以关注博主后续更新。

相关文章
|
3天前
|
搜索推荐 编译器 Linux
一个可用于企业开发及通用跨平台的Makefile文件
一款适用于企业级开发的通用跨平台Makefile,支持C/C++混合编译、多目标输出(可执行文件、静态/动态库)、Release/Debug版本管理。配置简洁,仅需修改带`MF_CONFIGURE_`前缀的变量,支持脚本化配置与子Makefile管理,具备完善日志、错误提示和跨平台兼容性,附详细文档与示例,便于学习与集成。
271 116
|
18天前
|
域名解析 人工智能
【实操攻略】手把手教学,免费领取.CN域名
即日起至2025年12月31日,购买万小智AI建站或云·企业官网,每单可免费领1个.CN域名首年!跟我了解领取攻略吧~
|
12天前
|
安全 Java Android开发
深度解析 Android 崩溃捕获原理及从崩溃到归因的闭环实践
崩溃堆栈全是 a.b.c?Native 错误查不到行号?本文详解 Android 崩溃采集全链路原理,教你如何把“天书”变“说明书”。RUM SDK 已支持一键接入。
663 219
|
5天前
|
数据采集 人工智能 自然语言处理
Meta SAM3开源:让图像分割,听懂你的话
Meta发布并开源SAM 3,首个支持文本或视觉提示的统一图像视频分割模型,可精准分割“红色条纹伞”等开放词汇概念,覆盖400万独特概念,性能达人类水平75%–80%,推动视觉分割新突破。
350 34
Meta SAM3开源:让图像分割,听懂你的话
|
10天前
|
人工智能 移动开发 自然语言处理
2025最新HTML静态网页制作工具推荐:10款免费在线生成器小白也能5分钟上手
晓猛团队精选2025年10款真正免费、无需编程的在线HTML建站工具,涵盖AI生成、拖拽编辑、设计稿转代码等多种类型,均支持浏览器直接使用、快速出图与文件导出,特别适合零基础用户快速搭建个人网站、落地页或企业官网。
1576 157
|
存储 人工智能 监控
从代码生成到自主决策:打造一个Coding驱动的“自我编程”Agent
本文介绍了一种基于LLM的“自我编程”Agent系统,通过代码驱动实现复杂逻辑。该Agent以Python为执行引擎,结合Py4j实现Java与Python交互,支持多工具调用、记忆分层与上下文工程,具备感知、认知、表达、自我评估等能力模块,目标是打造可进化的“1.5线”智能助手。
897 61
|
7天前
|
编解码 Linux 数据安全/隐私保护
教程分享免费视频压缩软件,免费视频压缩,视频压缩免费,附压缩方法及学习教程
教程分享免费视频压缩软件,免费视频压缩,视频压缩免费,附压缩方法及学习教程
295 140