leetcode-合并两个有序链表

news2025/1/23 22:25:35

目录

题目

图解

方法一

方法二

代码(解析在注释中)

方法一

​编辑方法二


题目

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

示例 1:

输入: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
  • l1 和 l2 均按 非递减顺序 排列

图解

方法一

最终效果

方法二

这个方法就比上一个方法多了一个“哨兵”,也就是用malloc开辟的一个辅助空间

代码(解析在注释中)

方法一

/**
 * 定义单链表结构体
 * 结构体中包含整数值val以及指向下一个节点的指针next
 */
typedef struct ListNode ListNode;
struct ListNode {
    int val;
    struct ListNode *next;
};

/**
 * 函数mergeTwoLists接收两个单链表(list1和list2)作为参数,
 * 并返回合并后的新链表,新链表中的元素按升序排列。
 */
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {
    // 首先判断输入的两个链表是否为空,如果其中一个为空,则直接返回另一个非空链表
    if (list1 == NULL) {
        return list2;
    }
    if (list2 == NULL) {
        return list1;
    }

    // 为了不对原链表进行修改,创建两个指针l1和l2分别指向list1和list2的头部
    ListNode* l1 = list1;
    ListNode* l2 = list2;

    // 初始化新链表的头结点和尾结点为NULL
    ListNode *Newhead, *Newtail;
    Newhead = Newtail = NULL;

    // 使用while循环遍历两个链表直到其中一个链表遍历完为止
    while (l1 && l2) {
        // 比较当前节点的值大小,将较小值的节点添加到新链表中
        if (l1->val < l2->val) {
            // 如果新链表还未添加过节点,则设置新链表的头结点和尾结点都为l1
            if (Newhead == NULL) {
                Newhead = Newtail = l1;
            } else {
                // 否则将尾结点的next指向l1,并更新尾结点为新添加的节点
                Newtail->next = l1;
                Newtail = Newtail->next;
            }
            // 移动l1指针至下一个节点
            l1 = l1->next;
        } else {
            // 类似地处理l2的情况
            if (Newhead == NULL) {
                Newhead = Newtail = l2;
            } else {
                Newtail->next = l2;
                Newtail = Newtail->next;
            }
            l2 = l2->next;
        }
    }

    // 当某一个链表遍历完之后,将未遍历完的链表剩余部分连接到新链表的尾部
    if (l1) {
        Newtail->next = l1;
    }
    if (l2) {
        Newtail->next = l2;
    }

    // 返回新链表的头结点
    return Newhead;
}

方法二

/**
 * 定义单链表结构体
 * 结构体中包含整数值val以及指向下一个节点的指针next
 */
typedef struct ListNode ListNode;
struct ListNode {
    int val;
    struct ListNode *next;
};

/**
 * 函数mergeTwoLists接收两个单链表(list1和list2)作为参数,
 * 合并这两个已排序的链表,并返回合并后的新链表,新链表中的元素仍按升序排列。
 *
 * 思路:
 * 1. 创建新的链表用于存放合并后的节点,初始化新链表头结点和尾结点。
 * 2. 使用while循环比较两个链表当前节点的值,将较小值的节点添加到新链表中。
 * 3. 当某个链表遍历完后,将另一个未遍历完链表的剩余部分添加到新链表尾部。
 * 4. 最后,释放初始分配给新链表头结点的空间,并返回新链表的第二个节点(实际内容的起始节点)。
 */
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {
    // 判断输入链表是否为空
    if (list1 == NULL) {
        return list2;
    }
    if (list2 == NULL) {
        return list1;
    }

    // 创建临时指针保存原始链表,避免改变它们
    ListNode* l1 = list1;
    ListNode* l2 = list2;

    // 分配内存创建新链表的头结点和尾结点
    ListNode *Newhead, *Newtail;
    Newhead = Newtail = (ListNode*)malloc(sizeof(ListNode));
    // 注意:这里实际上创建了一个空节点作为占位符,其next指针将指向实际的第一个合并节点

    // 循环遍历两个链表,将较小值的节点依次添加到新链表中
    while (l1 && l2) {
        if (l1->val < l2->val) {
            Newtail->next = l1;
            Newtail = Newtail->next;
            l1 = l1->next;
        } else {
            Newtail->next = l2;
            Newtail = Newtail->next;
            l2 = l2->next;
        }
    }

    // 将剩余未遍历完的链表连接到新链表尾部
    if (l1) {
        Newtail->next = l1;
    }
    if (l2) {
        Newtail->next = l2;
    }

    // 获取新链表的实际头部(即第一个有效节点),释放占位头结点的空间
    ListNode* next = Newhead->next;
    free(Newhead);
    Newhead = NULL; // 可选,置空便于调试或后续操作

    // 返回合并后的新链表的实际头部节点
    return next;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1600911.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

电机控制器电路板布局布线参考指导(五)

电机控制器电路板布局布线参考指导&#xff08;五&#xff09;大容量电容和旁路电容的放置 1.大容量电容的放置2.电荷泵电容器3.旁路电容/去耦电容的放置3.1 靠近电源3.2 靠近功率器件3.3 靠近开关电流源3.4 靠近电流感测放大器3.5 靠近稳压器 tips&#xff1a;资料主要来自网络…

Spring Boot 多环境配置:YML 文件的三种高效方法

&#x1f31f; 前言 欢迎来到我的技术小宇宙&#xff01;&#x1f30c; 这里不仅是我记录技术点滴的后花园&#xff0c;也是我分享学习心得和项目经验的乐园。&#x1f4da; 无论你是技术小白还是资深大牛&#xff0c;这里总有一些内容能触动你的好奇心。&#x1f50d; &#x…

【Redis 神秘大陆】006 灾备方案

六、Redis 灾备方案 6.1 存储方案 6.1.1 基础对比 RDB持久化AOF持久化原理周期性fork子进程生成持久化文件每次写入记录命令日志文件类型二进制dump快照文件文本appendonly日志文件触发条件默认超过300s间隔且有1s内超过1kb数据变更永久性每秒fsync一次文件位置配置文件中指…

基于Qt的二维码生成与识别

基于Qt的二维码生成与识别 一、获取QZxing开源库 1.通过封装的QZxing开源库生成和识别二维码&#xff0c;下载地址&#xff1a;GitCode - 开发者的代码家园https://gitcode.com/mirrors/ftylitak/qzxing/tree/master。 2.下载解压后&#xff0c;使用Qt Creator xx&#xff0…

如何采集opc服务器数据上传云端

为了进一步提高生产效率&#xff0c;生产制造的不断朝着智能化发展和升级&#xff0c;传统的自动化生产系统已经不能满足需求。传统的SCADA系统一般是用于现场的数据采集与控制&#xff0c;但是本地控制已经无法满足整个工厂系统智能化数字化的需求&#xff0c;智能化数字化是需…

NTC热敏电阻采集温度-单片机通用模板

NTC热敏电阻采集温度-单片机通用模板 一、NTC热敏电阻转换温度的原理二、AT104Tem.c的实现三、AT104Tem.h的实现 一、NTC热敏电阻转换温度的原理 ①NTC热敏电阻会随着温度的升高&#xff0c;电阻值R逐渐降低&#xff1b;②硬件搭建电阻分压电路采集ADC逆推热敏电阻当前的阻值&…

线上频繁fullgc问题-SpringActuator的坑

整体复盘 一个不算普通的周五中午&#xff0c;同事收到了大量了cpu异常的报警。根据报警表现和通过arthas查看&#xff0c;很明显的问题就是内存不足&#xff0c;疯狂无效gc。而且结合arthas和gc日志查看&#xff0c;老年代打满了&#xff0c;gc不了一点。既然问题是内存问题&…

[html]一个动态js倒计时小组件

先看效果 代码 <style>.alert-sec-circle {stroke-dasharray: 735;transition: stroke-dashoffset 1s linear;} </style><div style"width: 110px; height: 110px; float: left;"><svg style"width:110px;height:110px;"><cir…

新零售门店、商品、会员管理指标体系总览

新零售&#xff0c;旨在打破传统零售业的边界&#xff0c;引入先进科技和数字化手段&#xff0c;通过整合线上线下渠道&#xff0c;全面提升用户体验&#xff0c;并实现更智能、高效、个性化的零售运营模式。这一模式不仅仅关注销售产品&#xff0c;更注重构建全方位的购物生态…

SpringBoot整合minio服务

这里我选用的是JDK1.8 SpringBoot2.3.12.RELEASE 一、导入依赖 <dependency><groupId>io.minio</groupId><artifactId>minio</artifactId><version>8.2.2</version> </dependency> 二、导入工具类 注意&#xff1a;需要在…

eclipse中tomcat环境配置,2024年最新Web前端面试选择题

先自我介绍一下&#xff0c;小编浙江大学毕业&#xff0c;去过华为、字节跳动等大厂&#xff0c;目前阿里P7 深知大多数程序员&#xff0c;想要提升技能&#xff0c;往往是自己摸索成长&#xff0c;但自己不成体系的自学效果低效又漫长&#xff0c;而且极易碰到天花板技术停滞…

【位运算 子集状态压缩】982按位与为零的三元组

算法可以发掘本质&#xff0c;如&#xff1a; 一&#xff0c;若干师傅和徒弟互有好感&#xff0c;有好感的师徒可以结对学习。师傅和徒弟都只能参加一个对子。如何让对子最多。 二&#xff0c;有无限多1X2和2X1的骨牌&#xff0c;某个棋盘若干格子坏了&#xff0c;如何在没有坏…

IP协议如何进行地址管理?

如今&#xff0c;IP协议有两个版本&#xff0c;分别是IPv4和IPv6&#xff0c;IPv4是目前主要应用的版本。IPv4的IP地址是以4个字节的数字来表示的&#xff0c;比如 127.0.0.1。因此&#xff0c;IPv4所能表示IP地址的个数是2^32次方&#xff0c;也就是42亿多个&#xff0c;看起来…

当全连接队列满了,tcp客户端收到服务端RST信令的模拟

当tcp服务端全连接队列满了后&#xff0c;并且服务端也不accept取出连接&#xff0c;客户端再次连接时&#xff0c;服务端能够看到SYN_RECV状态。但是客户端看到的是ESTABLISHED状态&#xff0c;所以客户端自认为成功建立了连接&#xff0c;故其写往服务端写数据&#xff0c;发…

大数据平台搭建2024(三)

三&#xff1a;HBase安装 提前上传hbase安装包至虚拟机 1 上传、解压 tar -zxvf hbase-2.0.0-alpha2-bin.tar.gz -C /hadoop2 修改配置文件 在/hadoop/hbase-2.0.0-alpha2-bin/conf文件夹里 vi /hadoop/hbase-2.0.0-alpha2/conf/hbase-env.sh修改hbase-env.sh文件 export…

【单例模式】饿汉式、懒汉式、静态内部类--简单例子

单例模式是⼀个单例类在任何情况下都只存在⼀个实例&#xff0c;构造⽅法必须是私有的、由⾃⼰创建⼀个静态变量存储实例&#xff0c;对外提供⼀个静态公有⽅法获取实例。 目录 一、单例模式 饿汉式 静态内部类 懒汉式 反射可以破坏单例 道高一尺魔高一丈 枚举 一、单例…

【Conda基础命令】使用conda创建、查看、删除虚拟环境及可能的报错处理

文章目录 前言&#xff08;1&#xff09; 在默认路径下创建一个新的虚拟环境&#xff08;2&#xff09; 查看已有的虚拟环境&#xff08;3&#xff09; 删除已有的虚拟环境&#xff08;谨慎操作&#xff09;&#xff08;4&#xff09;激活虚拟环境&#xff08;5&#xff09;退出…

AI大模型日报#0416:李飞飞《2024年人工智能指数报告》、Sora加入Adobe、李彦宏聊百度大模型之路

​导读&#xff1a; 欢迎阅读《AI大模型日报》&#xff0c;内容基于Python爬虫和LLM自动生成。目前采用“文心一言”生成了每条资讯的摘要。标题: 刚刚&#xff0c;李飞飞团队发布《2024年人工智能指数报告》&#xff1a;10大趋势&#xff0c;揭示AI大模型的“喜”与“忧” 摘…

ubuntu 设置 root 用户密码,创建新用户并赋权限

ubuntu 设置 root 用户密码&#xff0c;创建新用户并赋权限 在适用于 Linux 的 Windows 子系统上运行 Linux GUI 应用&#xff0c; 安装 Ubuntu-20.04 系统&#xff0c;新安装好的系统&#xff0c;设置用户名密码时&#xff0c; root 用户密码默认为空&#xff0c;这时需要设置…

《系统架构设计师教程(第2版)》第9章-软件可靠性基础知识-05-软件可靠性测试

文章目录 1. 概述2. 定义软件运行剖面2.1 软件的使用行为建模2.2 输入域分层2.3 弧上的概率分配2.4 其他注意点 3. 可靠性测试用例设计4. 可靠性测试的实施4.1 测试前检查4.2 注意点4.2 可靠性测试的难点1&#xff09;失效判断的主观性2&#xff09;计算的错误结果不易被发现 4…