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

【LeetCode每日一题】——LCR 078.合并 K 个升序链表

文章目录

  • 一【题目类别】
  • 二【题目难度】
  • 三【题目编号】
  • 四【题目描述】
  • 五【题目注意】
  • 六【题目示例】
  • 七【题目提示】
  • 八【解题思路】
  • 九【时间频度】
  • 十【代码实现】
  • 十一【提交结果】

一【题目类别】

  • 优先队列

二【题目难度】

  • 困难

三【题目编号】

  • LCR 078.合并 K 个升序链表

四【题目描述】

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

五【题目注意】

  • 本题与主站 23 题相同: https://leetcode-cn.com/problems/merge-k-sorted-lists/

六【题目示例】

  • 示例 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 <= 10^4
  • 0 <= lists[i].length <= 500
  • -10^4 <= lists[i][j] <= 10^4
  • lists[i] 按 升序 排列
  • lists[i].length 的总和不超过 10^4

八【解题思路】

  • 使用优先队列(小顶堆)解决该问题
  • 小顶堆维护各个链表没有被合并的的节点的最前面的节点
  • 这样我们每次都会取出所有链表中值最小的节点
  • 然后依次将所有节点存入小顶堆中再将其合并为一个链表
  • 最后返回结果即可

九【时间频度】

  • 时间复杂度: O ( k n × l o g k ) O(kn × logk) O(kn×logk) k k k为优先队列中的元素个数, n n n为传入的链表个数
  • 空间复杂度: O ( k ) O(k) O(k) k k k为优先队列中的元素个数

十【代码实现】

  1. Java语言版
/*** Definition for singly-linked list.* public 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 {// 定义一个类 Status,用来存储链表节点值和对应的节点指针class Status implements Comparable<Status> {int val;ListNode node;// 构造函数,初始化节点值和指针Status(int val, ListNode node) {this.val = val;this.node = node;}// 实现 Comparable 接口,按照节点值从小到大排序public int compareTo(Status status) {return this.val - status.val;}}// 合并多个有序链表public ListNode mergeKLists(ListNode[] lists) {// 定义一个优先队列(小顶堆),会根据 Status 类中的节点值进行排序PriorityQueue<Status> pQueue = new PriorityQueue<Status>();// 遍历所有链表,把每个链表的第一个节点放入优先队列for (ListNode node : lists) {if (node != null) {pQueue.offer(new Status(node.val, node));}}// 创建一个虚拟头节点和尾节点,方便构建结果链表ListNode head = new ListNode(0);ListNode tail = head;// 循环处理优先队列中的节点,直到队列为空while (!pQueue.isEmpty()) {Status min_node = pQueue.poll();tail.next = min_node.node;tail = tail.next;if (min_node.node.next != null) {pQueue.offer(new Status(min_node.node.next.val, min_node.node.next));}}// 返回合并后的链表(哑节点的下一个节点)return head.next;}
}
  1. Python语言版
# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, val=0, next=None):
#         self.val = val
#         self.next = nextimport heapqclass Solution:# 定义一个类 Status,用来存储链表节点值和对应的节点指针class Status:# 构造函数,初始化节点值和指针def __init__(self, val, node):self.val = valself.node = node# 实现接口,按照节点值从小到大排序def __lt__(self, other):return self.val < other.val# 合并多个有序链表def mergeKLists(self, lists: List[ListNode]) -> ListNode:# 定义一个优先队列(小顶堆),会根据 Status 类中的节点值进行排序pQueue = []# 遍历所有链表,把每个链表的第一个节点放入优先队列for node in lists:if node is not None:heapq.heappush(pQueue, self.Status(node.val, node))# 创建一个虚拟头节点和尾节点,方便构建结果链表head = ListNode(0)tail = head# 循环处理优先队列中的节点,直到队列为空while pQueue:min_node = heapq.heappop(pQueue)tail.next = min_node.nodetail = tail.nextif min_node.node.next is not None:heapq.heappush(pQueue, self.Status(min_node.node.next.val, min_node.node.next))# 返回合并后的链表(哑节点的下一个节点)return head.next
  1. C++语言版
/*** 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 Status {public:int val;ListNode* node;Status(int v, ListNode* n) : val(v), node(n) {}bool operator<(const Status& other) const {return val > other.val;}
};class Solution {
public:ListNode* mergeKLists(vector<ListNode*>& lists) {std::priority_queue<Status> pQueue;for (ListNode* node : lists) {if (node != nullptr) {pQueue.push(Status(node->val, node));}}ListNode* head = new ListNode(0);ListNode* tail = head;while (!pQueue.empty()) {Status min_node = pQueue.top();pQueue.pop();tail->next = min_node.node;tail = tail->next;if (min_node.node->next != nullptr) {pQueue.push(Status(min_node.node->next->val, min_node.node->next));}}ListNode* result = head->next;delete head;return result;}
};

十一【提交结果】

  1. Java语言版
    在这里插入图片描述

  2. Python语言版
    在这里插入图片描述

  3. C++语言版
    在这里插入图片描述

http://www.lryc.cn/news/438131.html

相关文章:

  • 代码随想录算法训练营第五十九天 | dijkstra(堆优化版)精讲
  • go语言后端开发学习(七)——如何在gin框架中集成限流中间件
  • SpringBoot2:web开发常用功能实现及原理解析-整合EasyExcel实现Excel导入导出功能
  • CTFShow-信息搜集
  • Facebook的虚拟现实功能简介:社交网络的新前沿
  • Redis embstr 编码
  • 【Elasticsearch系列二】安装 Kibana
  • 中国电子学会202403青少年软件编程(Python)等级考试试卷(三级)真题与解析
  • k8s 资源管理
  • 演示:基于WPF的自绘的中国地铁轨道控件
  • 设计模式(Design Patterns)
  • C++:opencv生成结构元素用于膨胀腐蚀等cv::getStructuringElement
  • 最大余额法,解决百分比计算相加不等于100%(扇形/饼图百分比使用的此算法)
  • 哈希表简单介绍
  • kafka 之 本地部署单机版
  • 开发一款通过蓝牙连接控制水电表的微信小程序
  • 力扣14.最长公共前缀
  • 你也喜欢“钓鱼“吗?
  • druid jdbc 执行 sql 输出 开销耗时
  • C++如何处理内存碎片问题
  • FreeRTOS常用API接口函数
  • DesignPattern设计模式
  • 3.ChatGPT在教育领域的应用:教学辅助与案例分享(3/10)
  • Kafka+PostgreSql,构建一个总线服务
  • 电脑怎么录屏?四款录屏工具分享
  • Java代码审计篇 | ofcms系统审计思路讲解 - 篇4 | XXE漏洞审计
  • Leetcode 每日一题:Word Ladder
  • c++ 编辑器 和 编译器 的详细解释
  • 计算机视觉(二)—— MDPI特刊推荐
  • 交叉编译工具链的安装及带wiringPi库的交叉编译实现