【力扣】23. 合并 K 个升序链表 <链表指针、堆排序、分治>

目录

    • 【力扣】23. 合并 K 个升序链表
    • 题解
      • 方法一:暴力,先遍历取出来值到数组中排序,再生成新链表
      • 方法二:基础堆排序(使用优先队列 PriorityQueue)
      • 方法三:基础堆排序(使用优先队列 PriorityQueue)
      • 方法四:递归
      • 方法五:分治

【力扣】23. 合并 K 个升序链表

给你一个链表数组,每个链表都已经按升序排列。请你将所有链表合并到一个升序链表中,返回合并后的链表。

示例 1
输入:lists = [[1,4,5],[1,3,4],[2,6]]
输出:[1,1,2,3,4,4,5,6]

解释:
链表数组如下:
[ 1 ——> 4 ——> 5, 1——> 3 ——> 4, 2 ——> 6 ]
将它们合并到一个有序链表中得到。
1 ——> 1 ——> 2 ——> 3 ——> 4 ——> 4 ——> 5 ——> 6

示例 2
输入:lists = []
输出:[]

示例 3
输入:lists = [[]]
输出:[]

提示
k == lists.length
0 <= k <= 1 0 4 10^4 104
0 <= lists[i].length <= 500
- 1 0 4 10^4 104 <= lists[i][j] <= 1 0 4 10^4 104
lists[i] 按升序排列
lists[i].length 的总和不超过 1 0 4 10^4 104

题解

方法一:暴力,先遍历取出来值到数组中排序,再生成新链表

import java.util.*;

class ListNode {
    int val;
    ListNode next;
    ListNode() {}
    ListNode(int val) { this.val = val; }
    ListNode(int val, ListNode next) { this.val = val;this.next = next; }
}

class Solution {
    public ListNode mergeKLists(ListNode[] lists) {

        ListNode dummyNode = new ListNode(-1);
        ListNode prev = dummyNode;

		//先遍历取出来值到数组中排序
        List<Integer> nodes = new ArrayList();
        for(ListNode list: lists){
            while(list != null){
                nodes.add(list.val);
                list = list.next;
            }
        }
        Collections.sort(nodes);

		//生成新链表
        for(int x : nodes){
            prev.next = new ListNode(x);
            prev = prev.next;
        }

        return dummyNode.next;
    }
}

方法二:基础堆排序(使用优先队列 PriorityQueue)

思路:遍历数组每个值,建小顶堆,照着小顶堆依次取堆顶元素并移除,直到堆空

import java.util.*;

class ListNode {
    int val;
    ListNode next;
    ListNode() {}
    ListNode(int val) { this.val = val; }
    ListNode(int val, ListNode next) { this.val = val;this.next = next; }
}

class Solution {
    public ListNode mergeKLists(ListNode[] lists) {
        //边界
        if (lists == null || lists.length == 0) {
            return null;
        }

        //创建一个堆(优先队列),并设置元素的排序方式
        PriorityQueue<ListNode> queue = new PriorityQueue(new Comparator<ListNode>() {
            @Override
            public int compare(ListNode o1, ListNode o2) {
                return (o1.val - o2.val);
            }
        });

        //遍历链表数组,然后将每个链表的每个节点都放入堆中
        for(ListNode list: lists){
            while(list != null){
                queue.add(list);
                list = list.next;
            }
        }

        ListNode dummyNode = new ListNode(-1);
        ListNode prev = dummyNode;

        //从堆中不断取出元素,并将取出的元素串联起来
        while (!queue.isEmpty()) {
            prev.next = queue.poll();
            prev = prev.next;
        }
        prev.next = null;

        return dummyNode.next;
    }
}

方法三:基础堆排序(使用优先队列 PriorityQueue)

思路:只遍历每个数组第一个值(k 个),建小顶堆,照着小顶堆依次取堆顶元素并移除,移除的同时,如果这个值原来还有next 就补齐 k个到堆里,继续取堆顶移除。

因为:k 个链表中的最小值,一定来自 k 个递增链表中某一个的第一个值。将原先的 O(N) 的空间复杂度优化到 O(k)

import java.util.*;

class ListNode {
    int val;
    ListNode next;
    ListNode() {}
    ListNode(int val) { this.val = val; }
    ListNode(int val, ListNode next) { this.val = val;this.next = next; }
}

class Solution {
    public ListNode mergeKLists(ListNode[] lists) {
        //边界
        if (lists == null || lists.length == 0) {
            return null;
        }

        //创建一个小根堆,并定义好排序函数
        PriorityQueue<ListNode> queue = new PriorityQueue(new Comparator<ListNode>() {
            @Override
            public int compare(ListNode o1, ListNode o2) {
                return (o1.val - o2.val);
            }
        });

        //这里不再是一股脑全部放到堆中,而是只把 k 个链表的第一个节点放入到堆中
        for (ListNode list : lists) {
            ListNode eachHead = list;
            if (eachHead != null) {
                queue.add(eachHead);
            }
        }

        ListNode dummyNode = new ListNode(-1);
        ListNode prev = dummyNode;
        //之后不断从堆中取出节点,如果这个节点所在的链表还有下一个节点,就将下个节点也放入堆中
        while (queue.size() > 0) {
            ListNode node = queue.poll();
            prev.next = node;
            prev = prev.next;
            if (node.next != null) {
                queue.add(node.next);
            }
        }

        prev.next = null;
        return dummyNode.next;
    }
}

方法四:递归

合并两个链表的思路来合并 k 个链表

class ListNode {
    int val;
    ListNode next;
    ListNode() {}
    ListNode(int val) { this.val = val; }
    ListNode(int val, ListNode next) { this.val = val;this.next = next; }
}

class Solution {
    public ListNode mergeKLists(ListNode[] lists) {
        // 边界
        if (lists == null || lists.length == 0) {
            return null;
        }
        // 将 lists[0] 作为最终合并的链表,然后将 list[0] 和 lists[1] 合并成 lists[0-1]
        // 再将 lists[0-1] 和 lists[2] 合并,如此反复最终 lists[0] 就是最终结果
        ListNode res = lists[0];
        for (int i = 1; i < lists.length; i++) {
            res = merge(res, lists[i]);
        }

        return res;
    }

    // 合并两个有序链表,递归版本
    private ListNode merge(ListNode l1, ListNode l2) {
        //递归的结束条件,如果 l1 和 l2 中有一个为空就返回
        if (l1 == null || l2 == null) {
            return (l1 == null) ? l2 : l1;
        }

        //如果 l1 的值 <=l2 的值,就继续递归,比较 l1.next 的值和 l2 的值
        //l1.next 和 l2 比较完后,会产生一个更小的节点 x,将 x 加到当前 l1 的后面
        if (l1.val <= l2.val) {
            l1.next = merge(l1.next, l2);
            return l1;
        }
        //如果 l1 的值 >l2 的值,就继续递归,比较 l1 的值和 l2.next 的值
        else {
            l2.next = merge(l1, l2.next);
            return l2;
        }
    }
}

方法五:分治

  一开始数组的规模是 k,找到中间点,一分为二,然后再拆分,直到不能再拆分 (规模为1时) 时便返回。之后开始合并,合并的代码借用了合并两个排序链表的代码。
  当两个规模最小的链表合并完后,其规模就变大了,然后不断重复这个合并过程,直到最终得到一个有序的链表。
  分治就是不断缩小其规模,再不断合并扩大的过程

class Solution {
    public ListNode mergeKLists(ListNode[] lists) {
        // 边界
        if (lists == null || lists.length == 0) {
            return null;
        }
        //分治
        return helper(lists, 0, lists.length - 1);
    }


    //通过合并两个链表,不断增大其规模,整体看就是不断缩小-最后不断扩大的过程
    private ListNode helper(ListNode[] lists, int begin, int end) {
        if (begin == end) {
            return lists[begin];
        }

        //通过 mid 将数组一分为二,并不断缩小规模,当规模为 1 时返回并开始合并
        int mid = begin + (end - begin) / 2;
        ListNode left = helper(lists, begin, mid);
        ListNode right = helper(lists, mid + 1, end);

        return merge(left, right);
    }

    // 合并两个有序链表,递归版本
    private ListNode merge(ListNode l1, ListNode l2) {
        //递归的结束条件,如果 l1 和 l2 中有一个为空就返回
        if (l1 == null || l2 == null) {
            return (l1 == null) ? l2 : l1;
        }

        //如果 l1 的值 <=l2 的值,就继续递归,比较 l1.next 的值和 l2 的值
        //l1.next 和 l2 比较完后,会产生一个更小的节点 x,将 x 加到当前 l1 的后面
        if (l1.val <= l2.val) {
            l1.next = merge(l1.next, l2);
            return l1;
        }
        //如果 l1 的值 >l2 的值,就继续递归,比较 l1 的值和 l2.next 的值
        else {
            l2.next = merge(l1, l2.next);
            return l2;
        }
    }
}

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【深度学习笔记】深度学习框架

本专栏是网易云课堂人工智能课程《神经网络与深度学习》的学习笔记&#xff0c;视频由网易云课堂与 deeplearning.ai 联合出品&#xff0c;主讲人是吴恩达 Andrew Ng 教授。感兴趣的网友可以观看网易云课堂的视频进行深入学习&#xff0c;视频的链接如下&#xff1a; 神经网络和…

【深度学习】采用自动编码器生成新图像

一、说明 你知道什么会很酷吗&#xff1f;如果我们不需要所有这些标记的数据来训练 我们的模型。我的意思是标记和分类数据需要太多的工作。 不幸的是&#xff0c;大多数现有模型从支持向量机到卷积神经网&#xff0c;没有它们&#xff0c;卷积神经网络就无法训练。无监督学习不…

数据集的介绍及其标注

水到绝境是风景 人到绝境是重生 一、什么是目标检测 目标检测是计算机视觉领域的一个重要任务&#xff0c;旨在识别和定位图像或视频中的多个目标对象。与图像分类只关注图像属于哪个类别不同&#xff0c;目标检测不仅要确定目标所属的类别&#xff0c;还要准确地标记目标在图…

【Linux】冯诺伊曼体系结构|操作系统概念理解

个人主页&#xff1a;&#x1f35d;在肯德基吃麻辣烫 我的gitee&#xff1a;Linux仓库 个人专栏&#xff1a;Linux专栏 分享一句喜欢的话&#xff1a;热烈的火焰&#xff0c;冰封在最沉默的火山深处 文章目录 前言一、先谈硬件——冯诺依曼体系结构1.什么是冯诺依曼体系结构&am…

【Spring AOP】什么是AOP

文章目录 1、AOP思想2、AOP入门案例3、AOP工作流程4、AOP切入点表达式5、AOP的五种通知类型6、AOP通知获取数据7、案例&#xff1a;百度网盘密码数据兼容处理8、AOP总结 1、AOP思想 AOP&#xff0c;即Aspect Oriented Programming&#xff0c;面向切面编程。是一种编程范式&am…

Spring中的事务

一、为什么需要事务&#xff1f; 事务定义 将一组操作封装成一个执行单元&#xff08;封装到一起&#xff09;&#xff0c;要么全部成功&#xff0c;要么全部失败。 为什么要用事务&#xff1f; 比如转账分为两个操作&#xff1a; 第一步操作&#xff1a; A 账户 -100 元…

int[]数组转Integer[]、List、Map「结合leetcode:第414题 第三大的数、第169题 多数元素 介绍」

文章目录 1、int[ ] 转 Integer[ ]:2、两道leetcode题遇到的场景:2.1、int[ ] 转 List<Integer> :2.2、int[ ] 转 Map: 1、int[ ] 转 Integer[ ]: public static void main(String[] args) {int[] nums {1, 2, 3}; Integer[] array Arrays.stream(nums).boxed().to…

Qt 6. 其他类调用Ui中的控件

1. 把主类指针this传给其他类&#xff0c;tcpClientSocket new TcpClient(this); //ex2.cpp #include "ex2.h" #include "ui_ex2.h"Ex2::Ex2(QWidget *parent): QDialog(parent), ui(new Ui::Ex2) {ui->setupUi(this);tcpClientSocket new TcpClient…

51单片机(普中HC6800-EM3 V3.0)实验例程软件分析 实验二 LED闪烁

目录 前言 一、原理图及知识点介绍 二、代码分析 知识点四&#xff1a;delay(u16 i)这个函数为什么i1时&#xff0c;大约延时10us&#xff1f; 前言 已经是第二个实验了&#xff0c;上一个实验是点亮第一个LED灯&#xff0c;这个实验是LED的闪烁。 一、原理图及知识点介绍…

嵌入式开发学习(STC51-11-中断系统)

内容 外部中断-使用独立按键K3控制LED亮灭&#xff1b; 定时器中断-通过定时器0中断控制D1指示灯间隔1秒闪烁&#xff1b; 串口通信&#xff08;中断&#xff09;-通过串口&#xff08;UART&#xff09;实现与PC机对话&#xff0c;51单片机的串口收到PC机发来的数据后原封不动…

Cat.1如何成为物联网业务加速器?

随着Cat.1芯片及模组在功耗和成本上的不断优化&#xff0c;在窄带物联网领域&#xff0c;越来越多的终端客户把Cat.1当做与NB-IoT相比较的第二选择。越来越多的表计、烟感、市政等行业终端将Cat.1模组应用于非集中化部署的上报类终端业务中&#xff0c;Cat.1这只“网红猫”仍保…

动手学深度学习—深度学习计算(层和块、参数管理、自定义层和读写文件)

目录 1. 层和块1.1 自定义块1.2 顺序块1.3 在前向传播函数中执行代码 2. 参数管理2.1 参数访问2.1.1 目标参数2.1.2 一次性访问所有参数2.1.3 从嵌套块收集参数 2.2 参数初始化2.2.1 内置初始化2.2.2 自定义初始化 2.3 参数绑定 3. 自定义层3.1 不带参数的层3.2 带参数的层 4. …

数据安全治理实施办法

随着《数据安全法》和《个人信息保护法》陆续出台&#xff0c;各行业数据安全监管力度持续增强&#xff0c;数据安全合规成为企业数据安全治理体系建设的第一推动力。同时&#xff0c;企业普遍面临数据安全治理成本过高&#xff0c;对业务影响过大&#xff0c;实施路径不清晰等…

【雕爷学编程】Arduino动手做(186)---WeMos ESP32开发板14

37款传感器与模块的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&#x…

SQL从三个表中根据时间分别查询并汇总数量一行展示

需求&#xff1a;如果您要从三个表中根据时间分别查询并汇总数量&#xff0c;然后将结果以时间和数量一行展示&#xff0c;可以使用子查询和条件聚合。 入库主表 入库明细表 出库主表 出库明细表 退货主表 退货明细表 SQL代码 SELECT time,sum(a.inQty) as inQty,sum(a.outQty…

FFmpeg中硬解码后深度学习模型的图像处理dnn_processing(一)

ffmpeg 硬件解码 ffmpeg硬件解码可以使用最新的vulkan来做&#xff0c;基本上来说&#xff0c;不挑操作系统是比较重要的&#xff0c;如果直接使用cuda也是非常好的选择。 AVPixelFormat sourcepf AV_PIX_FMT_NV12;// AV_PIX_FMT_NV12;// AV_PIX_FMT_YUV420P;AVPixelFormat d…

O3DE的Pass

Pass介绍 Pass是具有输入和输出的渲染过程。 在最终渲染帧中看到的每个细节都是通过一系列Pass&#xff08;前一个Pass的输出是下一个Pass的输入&#xff09;计算出来的。Pass可以生成图像&#xff08;作为纹理、缓冲区或渲染目标&#xff09;。每个图像都包含关于场景的特定…

云曦暑期学习第四周——流量、日志分析

1 日志分析 1.1 What is 日志 日志&#xff0c;是作为记录系统与服务最直接有效的方法。在日志中&#xff0c;可以发现访问记录以及发现攻击线索。日志分析也是最常用的分析安全 事件所采用的途径。系统日志和 web 日志分别记录了不同内容&#xff0c;为分析攻击提供了有效证…

【敏捷开发】测试驱动开发(TDD)

测试驱动开发&#xff08;Test-Driven Development&#xff0c;简称TDD&#xff09;是敏捷开发模式中的一项核心实践和技术&#xff0c;也是一种设计方法论。TDD有别于以往的“先编码&#xff0c;后测试”的开发模式&#xff0c;要求在设计与编码之前&#xff0c;先编写测试脚本…

java+python企业会议在线办公微信小程序 ia505

一、小程序端功能 该部分内容提供员工注册、员工资料修改、通知公告、部门信息、会议记录等等功能。 二、管理员管理功能 该部分内容包含了首页、个人中心、通知公告管理、员工管理、部门信息管理、职位信息管理、会议记录管理、待办事项管理、工资信息管理、留言板管理、系统管…