算法通关村第一关——链表经典问题之合并有序链表三种方法一层一层优化

news2025/1/19 23:09:23

算法通关村第一关——链表经典问题之合并有序链表三种方法一层一层优化

题目描述

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

解题思路

第一种

新建一个链表,然后分别遍历两个链表,每次都按最小的节点接到新链表上,最后排完。

/** 第一个版本
     * @param list1
     * @param list2
     * @return 返回链表头部
     */
    public static LinkedNode mergeTwoLists(LinkedNode list1, LinkedNode list2) {
        LinkedNode newHead = new LinkedNode(-1);
        LinkedNode res = newHead;
        while (list1 != null || list2 != null) {
            // 情况1 : 都不为空的时候
            if (list1 != null && list2 != null) {
                if (list1.getData() < list2.getData()) {
                    newHead.setNext(list1);
                    list1 = list1.getNext();
                } else if (list1.getData() > list2.getData()) {
                    newHead.setNext(list2);
                    list2 = list2.getNext();
                } else { //相等的情况, 分别接两个链
                    newHead.setNext(list2);
                    list2 = list2.getNext();
                    newHead = newHead.getNext();
                    newHead.setNext(list1);
                    list1 = list1.getNext();
                }
                newHead = newHead.getNext();
                // 情况2 :加入还有链表一个不为空
            }else if(list1!=null && list2 == null){
                newHead.setNext(list1);
                list1 = list1.getNext();
                newHead = newHead.getNext();
            }else if(list1 == null && list2 !=null){
                newHead.setNext(list2);
                list2 = list2.getNext();
                newHead = newHead.getNext();
            }
        }
        return res.getNext();
    }

第二种

上面那版本的代码,看起来太臃肿了,虽然都能完成,但是所有处理都在一个大While循环里,我们可以将其变得苗条一些。第一个while只处理两个链表都不为空的情况,之后单独写while分别处理list1或者list2不为null的情况

/**
     * 改善一点点
     * @param list1
     * @param list2
     * @return 返回头节点
     */
    public static LinkedNode mergeTwoListsImprovement(LinkedNode list1,LinkedNode list2){
        LinkedNode newHead = new LinkedNode(-1);
        LinkedNode res = newHead;
        while(list1 != null && list2 != null){
            if(list1.getData() < list2.getData()){
                newHead.setNext(list1);
                list1 = list1.getNext();
            } else if (list1.getData() > list2.getData()){
                newHead.setNext(list2);
                list2 = list2.getNext();
            }else{
                newHead.setNext(list2);
                list2 = list2.getNext();
                newHead = newHead.getNext();
                newHead.setNext(list1);
                list1 = list1.getNext();
            }
            newHead = newHead.getNext();
        }
        // 下面的两个while 最多只有一个会执行
        while(list1 != null){
            newHead.setNext(list1);
            list1 = list1.getNext();
            newHead = newHead.getNext();
        }
        while(list2!=null){
            newHead.setNext(list2);
            list2 = list2.getNext();
            newHead = newHead.getNext();
        }

        return res.getNext();
    }

第三种

进一步分析,我们发现两个继续优化的点,一个是上面第一个大while里有三种情况,我们可以将其合并成两个,如果两个链表存在相同元素,第一次出现时使用if (l1.val <= l2.val)来处理,后面一次则会被else处理掉,什么意思呢?我们看一个序列。

假如list1为{1, 5, 8, 12},list2为{2, 5, 9, 13},此时都有一个node(5)。当两个链表都到5的位置时,出现了list1.val == list2.val,此时list1中的node(5)会被合并进来。然后list1继续向前走到了node(8),此时list2还是node(5),因此就会执行else中的代码块。这样就可以将第一个while的代码从三种变成两种,精简了很多。

第二个优化是后面两个小的while循环,这两个while最多只有一个会执行,而且由于链表只要将链表头接好,后面的自然就接上了,因此循环都不用写。

/**
     * 最终版
     * @param list1
     * @param list2
     * @return
     */
    public static LinkedNode mergeTwoListsOptimize(LinkedNode list1,LinkedNode list2){
        LinkedNode preHead = new LinkedNode(-1);
        LinkedNode prev = preHead;
        while(list1 != null && list2 != null){
            if(list1.getData() <= list2.getData()){
                prev.setNext(list1);
                list1 = list1.getNext();
            }else{
                prev.setNext(list2);
                list2 = list2.getNext();
            }
            prev = prev.getNext();
        }
        // 最多只有一个还没被合并完,直接上去就行了,这是链表合并方便的地方
        prev.setNext(list1 == null ? list2: list1);
        return preHead.getNext();
    }

整体代码

主代码

package AlgorithmFirst;

/**
 * 解决两个有序列表合并成一个有序列表
 */
public class MergeSortedLinedList {
    /** 第一个版本
     * @param list1
     * @param list2
     * @return 返回链表头部
     */
    public static LinkedNode mergeTwoLists(LinkedNode list1, LinkedNode list2) {
        LinkedNode newHead = new LinkedNode(-1);
        LinkedNode res = newHead;
        while (list1 != null || list2 != null) {
            // 情况1 : 都不为空的时候
            if (list1 != null && list2 != null) {
                if (list1.getData() < list2.getData()) {
                    newHead.setNext(list1);
                    list1 = list1.getNext();
                } else if (list1.getData() > list2.getData()) {
                    newHead.setNext(list2);
                    list2 = list2.getNext();
                } else { //相等的情况, 分别接两个链
                    newHead.setNext(list2);
                    list2 = list2.getNext();
                    newHead = newHead.getNext();
                    newHead.setNext(list1);
                    list1 = list1.getNext();
                }
                newHead = newHead.getNext();
                // 情况2 :加入还有链表一个不为空
            }else if(list1!=null && list2 == null){
                newHead.setNext(list1);
                list1 = list1.getNext();
                newHead = newHead.getNext();
            }else if(list1 == null && list2 !=null){
                newHead.setNext(list2);
                list2 = list2.getNext();
                newHead = newHead.getNext();
            }
        }
        return res.getNext();
    }

    /**
     * 改善一点点
     * @param list1
     * @param list2
     * @return 返回头节点
     */
    public static LinkedNode mergeTwoListsImprovement(LinkedNode list1,LinkedNode list2){
        LinkedNode newHead = new LinkedNode(-1);
        LinkedNode res = newHead;
        while(list1 != null && list2 != null){
            if(list1.getData() < list2.getData()){
                newHead.setNext(list1);
                list1 = list1.getNext();
            } else if (list1.getData() > list2.getData()){
                newHead.setNext(list2);
                list2 = list2.getNext();
            }else{
                newHead.setNext(list2);
                list2 = list2.getNext();
                newHead = newHead.getNext();
                newHead.setNext(list1);
                list1 = list1.getNext();
            }
            newHead = newHead.getNext();
        }
        // 下面的两个while 最多只有一个会执行
        while(list1 != null){
            newHead.setNext(list1);
            list1 = list1.getNext();
            newHead = newHead.getNext();
        }
        while(list2!=null){
            newHead.setNext(list2);
            list2 = list2.getNext();
            newHead = newHead.getNext();
        }

        return res.getNext();
    }

    /**
     * 最终版
     * @param list1
     * @param list2
     * @return
     */
    public static LinkedNode mergeTwoListsOptimize(LinkedNode list1,LinkedNode list2){
        LinkedNode preHead = new LinkedNode(-1);
        LinkedNode prev = preHead;
        while(list1 != null && list2 != null){
            if(list1.getData() <= list2.getData()){
                prev.setNext(list1);
                list1 = list1.getNext();
            }else{
                prev.setNext(list2);
                list2 = list2.getNext();
            }
            prev = prev.getNext();
        }
        // 最多只有一个还没被合并完,直接上去就行了,这是链表合并方便的地方
        prev.setNext(list1 == null ? list2: list1);
        return preHead.getNext();
    }

    public static void main(String[] args) {
        int[] val1 = {1,2,33,43,55,123};
        int[] val2 = {10,20,33,34,43,1234};
        LinkedNode list1 = LinkedNode.initList(val1);
        LinkedNode list2 = LinkedNode.initList(val2);
        // 这里要运行哪一个就开哪一个。
//        LinkedNode.printLinkedList(mergeTwoLists(list1,list2));
//        LinkedNode.printLinkedList(mergeTwoListsImprovement(list1,list2));
        LinkedNode.printLinkedList(mergeTwoListsOptimize(list1,list2));
    }
}

链表

package AlgorithmFirst;

public class LinkedNode {
    private int data;
    private LinkedNode next;

    public LinkedNode(int data) {
        this.data = data;
    }

    /**
     * 获取数据
     *
     * @return 数据值
     */
    public int getData() {
        return this.data;
    }

    /**
     * 设置数据的值
     *
     * @param data 数据
     */
    public void setData(int data) {
        this.data = data;
    }

    /**
     * 获取下一个节点
     *
     * @return 当前节点的下一个几点
     */
    public LinkedNode getNext() {
        return this.next;
    }

    /**
     * 设置下一个节点的值
     *
     * @param next 下一个节点
     */
    public void setNext(LinkedNode next) {
        this.next = next;
    }

    /**
     * 获取链表长度
     *
     * @param head 头节点
     * @return
     */
    public static int getListLength(LinkedNode head) {
        int length = 0;
        LinkedNode node = head;
        while (node != null) {
            length++;
            node = node.next;
        }
        return length;
    }

    /**
     * 缺省位置,直接在最后插入
     *
     * @param head       头节点
     * @param insertNode 插入节点
     * @return 头节点
     */
    public static LinkedNode insertNode(LinkedNode head, LinkedNode insertNode) {
        int size = getListLength(head);
        // return insertNode(head,insertNode,size+1); 修改一下,以便insertNode后面的元素能够全部插入进来。
        int count = 1;
        LinkedNode temp = head;
        if (head == null) {
            return insertNode;
        }
        while (temp != null) {
            if (count == size) {
                temp.next = insertNode;
                temp = null;
            } else {
                temp = temp.next;
                count++;
            }
        }
        return head;
    }

    /**
     * 指定位置插入
     *
     * @param head       头节点
     * @param nodeInsert 插入节点
     * @param position   插入位置,从1开始
     * @return 返回头节点
     */
    public static LinkedNode insertNode(LinkedNode head, LinkedNode nodeInsert, int position) {
        if (head == null) {
            // 如果head == null 表示当前链表为空,可以直接返回当前节点,或者报异常,这里直接把它当作头节点。
            return nodeInsert;
        }
        // 已经存在的元素的个数
        int size = getListLength(head);
        if (position > size + 1 || position < 1) {
            System.out.println("位置参数越界");
            return head;
        }

        // 表头插入
        if (position == 1) {
            nodeInsert.next = head;
            // 这里可以直接 return nodeInsert
            head = nodeInsert;
            return head;
        }

        LinkedNode pNode = head;
        int count = 1;
        // 这里position 被上面的size限制住了,不用考虑pNode = null
        while (count < position - 1) {
            pNode = pNode.next;
            count++;
        }
        nodeInsert.next = pNode.next;
        pNode.next = nodeInsert;
        return head;
    }

    /**
     * 缺省参数的删除最后一个节点
     *
     * @param head 链表头节点
     * @return 返回新链表头节点
     */
    public static LinkedNode deleteNode(LinkedNode head) {
        int size = getListLength(head);
        return deleteNode(head, size);
    }

    /**
     * 根据位置删除节点
     *
     * @param head     链表头节点
     * @param position 位置从1开始,最大链表大小 超出不删除,返回原头节点。
     * @return 新链表头节点
     */
    public static LinkedNode deleteNode(LinkedNode head, int position) {
        if (head == null) {
            // 链表为空,无法删除
            return null;
        }
        int size = getListLength(head);
        if (position > size || position < 1) {
            System.out.println("输入参数有误");
            return head;
        }
        if (position == 1) {
            return head.next;
        } else {
            LinkedNode cur = head;
            int count = 1;
            while (count < position - 1) {
                cur = cur.next;
                count++;
            }
            LinkedNode curNode = cur.next;
            cur.next = curNode.next;
            //上面两行可以简化成 : cur.next = cur.next.next
        }
        return head;
    }

    public static LinkedNode initList(int[] vals) {
        LinkedNode head = null;
        for (int val : vals) {
            head = insertNode(head, new LinkedNode(val));
        }
        return head;
    }

    public static void printLinkedList(LinkedNode head) {
        int count = 0;
        while (head != null) {
            System.out.println("第 " + ++count + " 个:" + head.data);
            head = head.next;
        }
    }

    public static void main(String[] args) {
        LinkedNode head = new LinkedNode(0);
        for (int i = 0; i < 10; i++) {
            head = LinkedNode.insertNode(head, new LinkedNode(i + 1));
        }
        System.out.println("origin:");
        printLinkedList(head);
        head = deleteNode(head, 3);
        System.out.println("delete the third ");
        printLinkedList(head);
        head = deleteNode(head);
        System.out.println("delete the last one");
        printLinkedList(head);
        head = insertNode(head, new LinkedNode(11));
        System.out.println("insert one from last");
        printLinkedList(head);
        head = insertNode(head, new LinkedNode(22222), 1);
        System.out.println("insert to first");
        printLinkedList(head);
    }

}

近期在自学 Java 做项目,加入了一个编程学习圈子,里面有编程学习路线和原创的项目教程,感觉非常不错。还可以 1 对 1 和大厂嘉宾交流答疑,也希望能对大家有帮助,扫 ⬇️ 二维码即可加入。

在这里插入图片描述

也可以点击链接:我正在「编程导航」和朋友们讨论有趣的话题,你⼀起来吧?

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

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

相关文章

记一次edu站点并拿下的过程cnvd

0x01 jeecg-boot介绍 JeecgBoot是一款基于代码生成器的低代码开发平台&#xff0c;零代码开发&#xff01;采用前后端分离架构&#xff1a;SpringBoot2.x&#xff0c;Ant Design&Vue&#xff0c;Mybatis-plus&#xff0c;Shiro&#xff0c;JWT。强大的代码生成器让前后端代…

k8s-17 k8s调度

调度器通过 kubernetes 的 watch 机制来发现集群中新创建且尚未被调度到 Node上的 Pod。调度器会将发现的每一个未调度的 Pod 调度到一个合适的 Node 上来运行。 kube-scheduler 是 Kubernetes 集群的默认调度器&#xff0c;并且是集群控制面的一部分如果你真的希望或者有这方面…

python如何连接数据库 ?一文详解pymysql的用法 。

使用Python连接数据库是常用的操作 &#xff0c;那么在Python代码中取操作数据库呢 &#xff1f; 接下来介绍一个包 &#xff1a;pymysql .它能帮我们在代码中连接MySQL数据库进行各种操作。 1.常用数据库模块 在做自动化测试时&#xff0c;我们经常会查库的需求 &#xff0c;…

解剖—顺序表相关OJ练习题

目录 一、删除有序数组中的重复项&#xff0c;返回出现一次元素的个数。 二、原地移除数组中所有数值等于val的元素 三、合并两个有序数组 四、旋转数组 五、数组形式的整数加法 一、删除有序数组中的重复项&#xff0c;返回出现一次元素的个数。 26. 删除有序数组中的重…

Service Mesh和Kubernetes:加强微服务的通信与安全性

文章目录 什么是Service Mesh&#xff1f;Service Mesh的优势1. 流量控制2. 安全性3. 可观测性 Istio&#xff1a;Service Mesh的领军者流量管理安全性可观测性 Linkerd&#xff1a;轻量级Service Mesh流量管理安全性可观测性 Istio vs. Linkerd实际应用结论 &#x1f388;个人…

vscode中4个json的区别和联系

在vscode中快捷键ctrlshiftp&#xff0c;然后输入setting&#xff0c;会出现下图几个选项 当不同设置之间出现冲突时&#xff0c;听谁的&#xff1a; Open Workspace Settings(JSON) > Open Settings(JSON) Open User Settings > Open Default Settings(JSON) Open Wo…

openstack 云主机 linux报 login incorrect

还未输入密码就提示login incorrect 不给输密码位置 完全不给输密码的机会 关机进入单用户 检查登录安全记录 vi /var/log/secure 发现 /usr/lib64/security/pam_unix.so 报错 将正常的机器提取/usr/lib64/security/pam_unix.so 比对MD5一致&#xff0c; 另外判断 libtir…

车载开发学习——CAN总线

CAN总线又称为汽车总线&#xff0c;全程为“控制器局域网&#xff08;Controller Area Network&#xff09;”&#xff0c;即区域网络控制器&#xff0c;它将区域内的单一控制单元以某种形式连接在一起&#xff0c;形成一个系统。在这个系统内&#xff0c;大家以一种大家都认可…

市值缩水90%以上,泛生子何以败退美股?

癌症是人类面临的最大健康威胁之一&#xff0c;也是医学界最难攻克的难题之一。随着科技的发展&#xff0c;癌症精准医疗逐渐成为治疗癌症的新方向&#xff0c;癌症精准医疗能通过对癌细胞的基因检测和分析&#xff0c;为患者提供个性化的治疗方案。然而&#xff0c;这一领域的…

redis(其它操作、管道)、django中使用redis(通用方案、 第三方模块)、django缓存、celery介绍(celery的快速使用)

1 redis其它操作 2 redis管道 3 django中使用redis 3.1 通用方案 3.2 第三方模块 4 django缓存 5 celery介绍 5.1 celery的快速使用 1 redis其它操作 delete(*names) exists(name) keys(pattern*) expire(name ,time) rename(src, dst) move(name, db)) randomkey() type(na…

VBA技术资料MF71:查找所有空格并替换为固定字符

我给VBA的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高自己的工作效率&#xff0c;而且可以提高数据的准确度。我的教程一共九套&#xff0c;分为初级、中级、高级三大部分。是对VBA的系统讲解&#xff0c;从简单的入门&#xff0c;到…

实现实时美颜:主播直播美颜SDK的技术细节

在今天的数字时代&#xff0c;直播和实时互动成为了日常生活的一部分&#xff0c;而主播直播美颜SDK的出现为用户提供了更加精美的视觉体验。这项技术的背后有着复杂的技术细节&#xff0c;从图像处理到机器学习&#xff0c;本文将深入探讨主播直播美颜SDK的技术细节&#xff0…

四边形不等式

区间dp问题&#xff0c;状态转移方程&#xff1a; dp[i][j] min( dp[i][k] dp[k1][j] w[i][j] ) //w[i][j]是从i到j的&#xff0c;一个定值 不随k改变&#xff0c;而且w的值只和i j有关&#xff0c;是它们的二元函数。 其中i<k<j ,初始值dp[i][i]已知。 含义&#x…

第三类医疗器械经营许可证经营范围

在我国&#xff0c;医疗器械监督管理条例规定:医械经营企业要依据主营产品办理相应许可证。医疗器械根据其风险性又分为三类&#xff0c;一类医疗器械实行产品备案管理&#xff0c;第二类、第三类医疗器械实行产品注册管理&#xff0c;经营第二类、第三类医疗器械应当持有《医疗…

Day 2 Qt

#include "my_widget.h" #include "ui_my_widget.h"My_Widget::My_Widget(QWidget *parent): QWidget(parent), ui(new Ui::My_Widget) {ui->setupUi(this);//窗口的相关设置 // this -> resize(800,500);this -> setWindowTitle("QQ聊天…

APP备案公钥、证书MD5指纹/签名MD5值获取方法

本文只详细讲解android app获取方法&#xff0c;三种方式&#xff1a; 1. 你的应用已安装到手机&#xff0c;android应用市场搜索下载安装 APP备案助手&#xff0c;此app可直接获取所有已安装app的公钥、证书MD5指纹/签名MD5值&#xff0c;示例&#xff1a;获取 抖音app公钥、…

每日一练 | 华为认证真题练习Day120

1、MPLS域中的LER全称为Label Egress Router。 A. 对 B. 错 2、如果一个以太网数据帧的Type/Length字段的值为0x0800&#xff0c;则此数据帧所承载的上层报文首部长度范围为20-60B。 A. 对 B. 错 3、在VRP平台上&#xff0c;可以通过下面哪种方式访问上一条历史命令&#x…

AI爆文变现-写作项目-脚本配置教程-解放双手

之前给大家分享过AI爆文的写作教程&#xff0c;没看过的可以看下对应的教程&#xff1a; AI爆文撸流量主保姆级教程2.0 因为是怼量&#xff0c;为了高效完成文章&#xff0c;我用python脚本实现了自动写文章的功能&#xff0c;发布文章目前还是要手动进行。 AI爆文教程参考&…

C++11——包装器与lambda表达式

目录 一.背景 二.lambda 1.见一见lambda 2.lambda表达式语法 3.lambda捕捉列表说明 三.函数对象与lambda表达式 四.包装器 1.function包装器 2.包装类的成员函数 五.bind 1.调整参数位置 2.减少函数参数 一.背景 在C98中&#xff0c;如果想要对一个数据集合中的元素…

c++_learning-模板元编程

模板元编程 元编程&#xff1a;元函数&#xff1a;定义&#xff1a;数值元函数&#xff1a;编译期间&#xff0c;能够被调用的类模板&#xff1a;constexpr修饰的函数&#xff1a;constexpr修饰的变量模板&#xff1a; 类型元函数&#xff1a;元函数总结&#xff1a; 混合元编程…