当前位置: 首页 > news >正文

排序(4)-归并排序专题——归并排序的分治美学

📌 引言

如果说快速排序是一位大开大合的“开拓者”,那归并排序(Merge Sort)就是一位步步为营的“哲学家”。它完美践行了计算机科学中最核心的分治思想(Divide and Conquer)。同时,它因为极其优秀的稳定性,成为了许多语言(包括 Java)实现工业级排序算法的绝对基石。

1. 归并排序(Merge Sort)

💡 核心思想

归并排序的核心可以用八个字概括:先分后治,合二为一

  • 分(Divide):找到数组的中点,将其一分为二。递归地对左半部分和右半部分继续切分,直到子数组的长度为 1(长度为 1 的数组天然有序)。

  • 治/合(Merge):将两个已经有序的子数组合并成一个更大的、完全有序的数组。在这个过程中需要一个辅助数组来暂存数据。

💻 Java 代码实现(高性能标准版)
public class MergeSort { public static void sort(int[] arr) { if (arr == null || arr.length < 2) return; // 关键优化:预先分配好辅助空间,避免在递归方法中频繁 new 数组导致内存碎片 int[] aux = new int[arr.length]; sort(arr, 0, arr.length - 1, aux); } private static void sort(int[] arr, int left, int right, int[] aux) { if (left >= right) return; // 递归基:子数组长度为 1 int mid = left + (right - left) / 2; // 防溢出的中点写法 sort(arr, left, mid, aux); // 递归使左半部分有序 sort(arr, mid + 1, right, aux); // 递归使右半部分有序 // 优化:如果左边的最大值已经小于等于右边的最小值,说明已经整体有序,无需 merge if (arr[mid] > arr[mid + 1]) { merge(arr, left, mid, right, aux); // 核心合并操作 } } private static void merge(int[] arr, int left, int mid, int right, int[] aux) { // 1. 将当前区间的元素复制到辅助数组 for (int k = left; k <= right; k++) { aux[k] = arr[k]; } // 2. 双指针合并 int i = left; // 左半部分指针 int j = mid + 1; // 右半部分指针 for (int k = left; k <= right; k++) { if (i > mid) { arr[k] = aux[j++]; // 左半边已用尽,直接取右半边 } else if (j > right) { arr[k] = aux[i++]; // 右半边已用尽,直接取左半边 } else if (aux[i] <= aux[j]) { arr[k] = aux[i++]; // 左边小或相等,取左边(这里的 <= 保证了算法的稳定性) } else { arr[k] = aux[j++]; // 右边小,取右边 } } } }
📊 性能分析
  • 时间复杂度:无论数组初始状态如何,归并排序的递归树层数都是 \log n,每层合并的开销都是 O(n)。因此,它的最好、最坏、平均时间复杂度全都是严格的O(n \log n)

  • 空间复杂度:O(n)。由于在合并阶段需要和原数组等长的辅助数组来暂存元素,这是它相比于快排和堆排序最大的劣势(空间换时间)。

  • 稳定性:稳定。只要在merge逻辑中,当左右指针元素相等时优先采用左侧元素(aux[i] <= aux[j]),就能完美保持相同元素的相对顺序。

http://www.cnnetsun.cn/news/2887580.html

相关文章:

  • 保姆级教程:手把手教你用ABAP查询T001B表,精准判断日期是否在OB52财务账期内
  • 从SPI Mode0/3时序图到PCB走线:高频SPI稳定性的‘隐形杀手’与避坑指南
  • vLLM 云原生推理基础设施深度解析:从 PagedAttention 内核到 Kubernetes 生产级部署
  • 别再只防外网了!用DHCP Snooping+IPSG给你的内网接入层加把‘锁’
  • 别再只点灯了!树莓派Pico的PWM信号详解:如何精准控制舵机角度与速度
  • DFT面积与性能的权衡:手把手教你根据项目需求选择Shared还是Dedicated Wrapper Cell
  • 避坑指南:若依多用户登录中Spring Security的Bean冲突与权限隔离陷阱
  • 第十二章 常用类
  • Quickshell技术架构解析:QtQuick桌面环境构建的艺术与工程
  • i.MX6ULL平台libmodbus 3.1.6交叉编译实操资源包(含补丁说明与完整构建脚本)
  • Claude Mythos:AI原生安全引擎如何重构漏洞挖掘范式
  • 别让你的SPI Nor跑飞了!100MHz高频下采样延时到底该怎么配?(附XTX芯片实测)
  • 德国法院裁决:谷歌需为 AI 概述虚假陈述负责,或影响全球 AI 搜索引擎
  • 从Hard Label到Soft Label:深入解析Label Smoothing的数学之美与实战调优
  • 如何5秒解锁百度网盘加密资源:智能提取码解析终极指南
  • 如何降低谷歌广告CPC?中小企业常用的低成本方法
  • League Akari:5个智能功能彻底改变你的英雄联盟游戏体验
  • 拓扑透镜的时间延迟公式严格推导(世毫九IGP框架)
  • 永磁同步电机静止状态下用方波注入法估算转子初始位置的Simulink仿真模型
  • PotPlayer百度翻译插件:5分钟搞定免费字幕实时翻译的终极指南
  • 从TIM1到TIM1.5:芯片封装散热设计的范式转移与技术对比
  • 平衡车项目实战:用STM32F103的EXTI中断实时读取MPU6050数据(附完整工程)
  • Vivado工程版本升级中IP缓存状态异常解析:从“Using cached IP results”到“synth_design Complete!”的实战处理
  • STM32F103 USB开发避坑指南:为什么你的端点数据会“神秘消失”?详解BTABLE与缓冲区地址计算
  • Android NDK原生层黑白滤镜实时预览方案(Camera2+OpenGL FBO)
  • C语言链表实战:从零手搓一个学生信息管理系统(附完整源码与内存管理避坑指南)
  • UniShare框架:社交分享场景下的联合推荐技术解析
  • 从‘显示一张地图’到‘定制你的地图’:OpenLayers 7.x 核心四要素实战拆解
  • 上岸必看!【中药学】必背100题及解析(卷号:06111014_07)
  • 杰理之U盘播放无损格式音频导致杰理之家的文件浏览线程运行加载文件信息很慢【篇】