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

深入理解归并排序:分治艺术的经典实践

归并排序(Merge Sort)是分治算法思想的经典体现,凭借稳定的 O(n log n) 时间复杂度,成为处理大规模数据的可靠选择。本文将从算法原理到链表实现,全面解析这一优雅的排序算法。


一、算法流程解析

1. 分治策略(Divide and Conquer)

  • 分解阶段:递归地将当前数组拆分为两个子数组,直至每个子数组只剩单个元素

  • 解决阶段:对最小单元(单元素数组)天然有序的情况进行处理

  • 合并阶段:将两个已排序的子数组合并为一个有序数组

2. 合并过程详解

function merge(left: number[], right: number[]): number[] {
  const result: number[] = [];
  let i = 0, j = 0;
  
  // 双指针遍历比较
  while (i < left.length && j < right.length) {
    if (left[i] <= right[j]) {
      result.push(left[i++]);
    } else {
      result.push(right[j++]);
    }
  }
  
  // 处理剩余元素
  return result.concat(left.slice(i)).concat(right.slice(j));
}

二、算法特性分析

1. 时间复杂度

  • 最佳/最差/平均时间复杂度均为 O(n log n)

  • 推导公式:T(n) = 2T(n/2) + O(n)

2. 空间复杂度

  • 数组实现:O(n) 临时存储空间

  • 链表实现:O(1) 原地修改指针(递归栈空间除外)

3. 稳定性

  • 稳定排序算法(合并时优先选择左半部分元素)

4. 适用场景

  • 大数据量排序

  • 链表排序

  • 需要稳定性的场景


三、链表排序优势

相比数组实现,归并排序在链表排序中展现独特优势:

  1. 空间优化:合并过程只需修改指针,无需额外存储空间

  2. 天然适配:链表节点离散存储,避免数组的随机访问需求

  3. 高效拆分:快慢指针法可在 O(n) 时间复杂度内找到链表中点


四、TypeScript 链表实现

class ListNode {
  constructor(public val: number = 0, public next: ListNode | null = null) {}
}

// 合并两个有序链表
function mergeLists(l1: ListNode | null, l2: ListNode | null): ListNode | null {
  const dummy = new ListNode();
  let current = dummy;
  
  while (l1 && l2) {
    if (l1.val <= l2.val) {
      current.next = l1;
      l1 = l1.next;
    } else {
      current.next = l2;
      l2 = l2.next;
    }
    current = current.next;
  }
  
  current.next = l1 || l2;
  return dummy.next;
}

// 归并排序主函数
function mergeSort(head: ListNode | null): ListNode | null {
  if (!head?.next) return head;
  
  // 快慢指针找中点
  let slow: ListNode = head;
  let fast: ListNode | null = head.next;
  
  while (fast?.next) {
    slow = slow.next!;
    fast = fast.next.next;
  }
  
  // 分割链表
  const mid = slow.next;
  slow.next = null;
  
  // 递归排序
  const left = mergeSort(head);
  const right = mergeSort(mid);
  
  return mergeLists(left, right);
}

五、使用示例

// 创建测试链表:4 -> 2 -> 1 -> 3
const head = new ListNode(4);
head.next = new ListNode(2);
head.next.next = new ListNode(1);
head.next.next.next = new ListNode(3);

// 执行排序
const sorted = mergeSort(head);

// 输出结果:1 -> 2 -> 3 -> 4
let current = sorted;
while (current) {
  console.log(current.val + ' -> ');
  current = current.next;
}

六、总结

归并排序通过分治策略将复杂问题简单化,其稳定性和可预测的时间复杂度使其成为:

  • 大数据量场景的优选算法

  • 链表排序的标准解决方案

  • 外部排序(如大文件处理)的核心组件

如果对你有帮助,请帮忙点个赞 

http://www.dtcms.com/a/106653.html

相关文章:

  • 【AI产品分享】面向图片的原始位置翻译功能
  • Redisson中BitMap位图的基本操作
  • CORS与OPTIONS请求
  • 蓝桥杯 游戏 6251 单调队列
  • .NET 创建MCP使用大模型对话
  • 【计网速通】计算机网络核心知识点与高频考点——数据链路层(二)
  • kafka消息可靠性传输语义
  • 大语言模型开发框架——LangChain
  • 使用PyTorch实现LeNet-5并在Fashion-MNIST数据集上训练
  • 【Linux】内核驱动学习笔记(二)
  • 基于Spring AI与Ollama构建本地DeepSeek对话机器人
  • 数据库分库分表中间件及对比
  • ensp 网络模拟器 思科华为基于VLANIF的公司网络搭建
  • 2025.4.2总结
  • Go语言GC:三色标记法工程启示|Go语言进阶(3)
  • K-means算法
  • 从零搭建微服务项目Pro(第7-1章——分布式雪花算法)
  • cmake(11):list 选项 排序 SORT,定义宏 add_definitions,cmake 里预定义的 8 个宏
  • Git 命令大全:通俗易懂的指南
  • 基于大模型预测风湿性心脏病二尖瓣病变的多维度诊疗研究报告
  • 内网隔离环境下Java实现图片预览的三大解决方案
  • 【Django开发】前后端分离django美多商城项目第15篇:商品搜索,1. Haystack介绍和安装配置【附代码文档】
  • 从 ZStack 获取物理机与云主机信息并导出 Excel 文件
  • visual studio 2022的windows驱动开发
  • C# System.Text.Json 中 JsonIgnoreCondition 使用详解
  • Linux2 CD LL hostnamectl type mkdir dudo
  • 跨系统平台实践:在内网自建kylin服务版系统yum源
  • 面基JavaEE银行金融业务逻辑层处理金融数据类型BigDecimal
  • AI提示词:好评生成器
  • 鸿蒙NEXT小游戏开发:数字华容道