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

LeetCode-链表-合并两个有序链表

image-20250520203051704

LeetCode-链表-合并两个有序链表

✏️ 关于专栏:专栏用于记录 prepare for the coding test


文章目录

  • LeetCode-链表-合并两个有序链表
    • 📝 合并两个有序链表
      • 🎯题目描述
      • 🔍 输入输出示例
      • 🧩题目提示
      • 🧪AC递归
      • 🧪AC迭代
    • 🌟 总结

📝 合并两个有序链表

🎯题目描述

将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。

🔗题目链接:合并两个有序链表

🔍 输入输出示例

示例 1:

img
输入:l1 = [1,2,4], l2 = [1,3,4]
输出:[1,1,2,3,4,4]

示例 2:

输入:l1 = [], l2 = []
输出:[]

示例 3:

输入:l1 = [], l2 = [0]
输出:[0]

🧩题目提示

  • 两个链表的节点数目范围是 [0, 50]
  • -100 <= Node.val <= 100
  • l1l2 均按 非递减顺序 排列

🧪AC递归

可以直接将 mergeTwoLists 用作递归函数来合并两个有序链表:

  • 递归终止条件:当其中一个链表为空时,说明不需要再继续合并,直接返回另一个非空链表即可,这个链表本身就是有序的。
  • 递归处理过程:当两个链表都不为空时,比较当前节点的值。若 list1 的当前节点值较小,则将 list1.nextlist2 继续递归合并,并将合并后的结果接在 list1 当前节点后面,返回 list1。反之,则将 list2.nextlist1 合并,并将结果接到 list2 后,最后返回 list2

这种方式通过递归自然地完成了节点的选择与链接,实现了有序合并。

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode() : val(0), next(nullptr) {}*     ListNode(int x) : val(x), next(nullptr) {}*     ListNode(int x, ListNode *next) : val(x), next(next) {}* };*/
class Solution {
public:ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {if(list1 == nullptr) return list2;if(list2 == nullptr) return list1;if(list1->val < list2->val){list1->next = mergeTwoLists(list1->next,list2);return list1;}else{list2->next = mergeTwoLists(list1,list2->next);return list2;}}
};

🧪AC迭代

我们可以先创建一个哨兵节点,它位于最终合并链表的头节点之前。这样做的好处是能统一处理流程,不用单独处理头节点或考虑链表为空的特殊情况,整体逻辑更加清晰简洁。

接下来,遍历两个链表,比较当前节点的值。若 list1 当前节点的值较小,就将其接到新链表的尾部,并让 list1 向后移动一位;反之,则将 list2 的当前节点接上,并将 list2 向后推进。若两者值相同,我们可以统一选择将 list2 的节点链接到结果链表中。

这一过程会持续进行,直到 list1list2 中至少有一个遍历完毕。

当循环结束时,仍可能存在某个链表还有未处理完的节点。此时可以直接将剩下的部分追加到合并链表的末尾。

最终返回的是哨兵节点的 next 指针所指向的节点,也就是新链表的第一个有效节点。

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode() : val(0), next(nullptr) {}*     ListNode(int x) : val(x), next(nullptr) {}*     ListNode(int x, ListNode *next) : val(x), next(next) {}* };*/
class Solution {
public:ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {ListNode dummy{};   //用哨兵节点简化代码逻辑auto cur = &dummy;  //cur指向新链表的末尾while(list1&&list2){if(list1->val < list2->val){cur->next = list1;list1 = list1->next;}else{cur->next = list2;list2 = list2->next;}cur = cur->next;}cur->next = list1 ? list1 : list2;return dummy.next;}
};

🌟 总结

合并两个有序链表可以通过递归迭代两种方式来实现,思路都围绕“逐步选择较小值节点”展开。

递归法

  • 思想简洁,利用函数调用栈自动维护合并过程。
  • 每次比较两个链表当前节点,选出较小值节点作为当前结果节点,继续递归合并剩余部分。
  • 递归终止条件是任一链表为空,直接返回另一个链表。

迭代法

  • 更加实用,避免了递归带来的额外栈空间开销。
  • 使用一个哨兵(dummy)节点作为新链表的起点,便于处理头节点和边界情况。
  • 通过 cur 指针逐步向后链接较小节点,最后再拼接剩余部分。

技巧亮点

  • 哨兵节点(dummy)是处理链表问题中常见且高效的技巧,避免处理头节点的特殊逻辑。
  • 两种方法都充分利用了“链表本身有序”的性质,无需新建节点,仅通过指针重新组织结构即可。
http://www.lryc.cn/news/2383343.html

相关文章:

  • sqli-labs靶场29-31关(http参数污染)
  • 独占内存访问指令LDXR/STXR
  • JVM 垃圾回收机制深度解析(含图解)
  • 如何利用 Conda 安装 Pytorch 教程 ?
  • 【ffmpeg】SPS与PPS的概念
  • uniapp vue 开发微信小程序 分包梳理经验总结
  • 什么是VR展示?VR展示的用途
  • .NET外挂系列:4. harmony 中补丁参数的有趣玩法(上)
  • Go语言中new与make的深度解析
  • 3、ubantu系统 | 通过vscode远程安装并配置anaconda
  • 【Unity】 HTFramework框架(六十五)ScrollList滚动数据列表
  • 深度学习之用CelebA_Spoof数据集搭建一个活体检测-用MNN来推理时候如何利用Conan对软件包进行管理
  • React 常见的陷阱之(如异步访问事件对象)
  • Swagger在java的运用
  • 代码随想录算法训练营 Day49 图论Ⅰ 深度优先与广度优先
  • .NET外挂系列:1. harmony 基本原理和骨架分析
  • HarmonyOS NEXT端云一体化工程目录结构
  • Ajax研究
  • 学习 Android(十)Fragment的生命周期
  • flutter 常用组件详细介绍、屏幕适配方案
  • Elasticsearch生产环境性能调优指南
  • 野火鲁班猫(arrch64架构debian)从零实现用MobileFaceNet算法进行实时人脸识别(一)conda环境搭建
  • RT Thread FinSH(msh)调度逻辑
  • Kotlin 极简小抄 P9 - 数组(数组的创建、数组元素的访问与修改、数组遍历、数组操作、多维数组、数组与可变参数)
  • CSS display有几种属性值
  • 【后端】【UV】【Django】 `uv` 管理的项目中搭建一个 Django 项目
  • 单片机设计_四轴飞行器(STM32)
  • kafka配置SASL_PLAINTEXT简单认证
  • PostgreSQL简单使用
  • 【Spring Boot】配置实战指南:Properties与YML的深度对比与最佳实践