LeetCode 23 ——合并 K 个排序链表
1. 题目

2. 解答
2.1. 方法一
在 合并两个有序链表 的基础上,我们很容易想到第一种解法,首先我们将第一个链表和第二个链表合并成一个新的链表,然后再往后依次合并接下来的每个链表即可。
假设每个链表结点数一样都为 n,第一次合并时,要遍历 2n 个结点,往后则要分别遍历 3n, 4n, ... , kn 个结点。可以看到,每次进行合并时都要将之前所有的链表遍历一次,因此这个方法的时间复杂度较高,为 \(O((\frac{k(k+1)}{2} - 1) * n)\)。
代码如下
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/
class Solution {
public:
ListNode* mergeKLists(vector<ListNode*>& lists) {
int k = lists.size(); // 链表个数
if (k >= 2)
{
// 先将前两个链表合并为一个新链表,再把新链表依次和后面的链表合并
ListNode *head = mergeTwoLists(lists[0], lists[1]);
for (int i = 2; i < k; i++)
{
head = mergeTwoLists(head, lists[i]);
}
return head;
}
else if (k == 1)
{
return lists[0];
}
else
{
return NULL;
}
}
ListNode* mergeTwoLists(ListNode* l1, ListNode* l2) {
ListNode *head = new ListNode(0); // 新建哨兵结点,方便操作
ListNode *temp = head;
// 依次比较两个链表的结点值,将值较小的结点插入到新建的链表后面
while(l1 && l2)
{
if (l2->val <= l1->val)
{
temp->next = l2;
temp = temp->next;
l2 = l2->next;
}
else
{
temp->next = l1;
temp = temp->next;
l1 = l1->next;
}
}
// 其中一个链表比较完毕,将另外一个链表剩余结点直接插入到新建的链表后面
if (l1)
{
temp->next = l1;
}
else
{
temp->next = l2;
}
temp = head;
head = head->next;// 删除哨兵结点
delete(temp);
return head;
}
};
2.2. 方法二
我们还可以每次只合并相邻的两个链表,这样经过第一次合并后,链表个数减半,我们一直重复这个过程,直到最后剩余一个链表即可。
假设每个链表结点数一样都为 n,第一次合并时,要进行 \(\frac{k}{2}\) 次合并,每次合并要遍历 2n 个结点;第二次合并时,要进行 \(\frac{k}{4}\) 次合并,每次合并要遍历 4n 个结点;可见,每次合并过程都只需要遍历 kn 次结点。而总共要进行 \(log_2k\) 个循环,因此这个方法的时间复杂度为 \(O(n * klog_2k)\),比第一个改善了许多。
代码如下
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/
class Solution {
public:
ListNode* mergeKLists(vector<ListNode*>& lists) {
int k = lists.size(); // 链表个数
if (k >= 2)
{
while(k != 1) // 循环合并过程直到剩余一个链表
{
if (k % 2 == 0) // 总共偶数个链表,依次合并相邻两个链表然后从前往后放到 vector 中去
{
for (int i = 0; i < k; i=i+2)
{
lists[i / 2] = mergeTwoLists(lists[i], lists[i+1]);
}
k = k / 2; // 每次合并后的新链表个数
}
else // 总共奇数个链表,依次合并相邻两个链表然后从前往后放到 vector 中去,最后余一个链表直接放入 vector 最后面
{
for (int i = 0; i < k - 1; i=i+2)
{
lists[i / 2] = mergeTwoLists(lists[i], lists[i+1]);
}
lists[k / 2] = lists[k-1];
k = k / 2 + 1; // 每次合并后的新链表个数
}
}
return lists[0];
}
else if (k == 1)
{
return lists[0];
}
else
{
return NULL;
}
}
ListNode* mergeTwoLists(ListNode* l1, ListNode* l2) {
ListNode *head = new ListNode(0); // 新建哨兵结点,方便操作
ListNode *temp = head;
// 依次比较两个链表的结点值,将值较小的结点插入到新建的链表后面
while(l1 && l2)
{
if (l2->val <= l1->val)
{
temp->next = l2;
temp = temp->next;
l2 = l2->next;
}
else
{
temp->next = l1;
temp = temp->next;
l1 = l1->next;
}
}
// 其中一个链表比较完毕,将另外一个链表剩余结点直接插入到新建的链表后面
if (l1)
{
temp->next = l1;
}
else
{
temp->next = l2;
}
temp = head;
head = head->next;// 删除哨兵结点
delete(temp);
return head;
}
};
2.3. 方法三
利用 C++ 模板库中的优先队列构建小顶堆,每次首结点元素值最小的链表出队,然后将首结点插入到新链表后面,再把删除首结点后的子链表放入队列中继续比较,直到队列为空结束。
代码如下
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/
class Solution {
public:
ListNode* mergeKLists(vector<ListNode*>& lists) {
int k = lists.size();
struct cmp
{
bool operator()(ListNode* a, ListNode* b)
{
return a->val > b->val;
}
};
priority_queue<ListNode *, vector<ListNode*>, cmp> q; // 构建小顶堆
for (int i = 0; i < k; i++)
{
if (lists[i])
q.push(lists[i]); // 链表非空,加入队列
}
ListNode *head = new ListNode(0); // 新建哨兵结点,方便操作
ListNode *temp = head;
while(!q.empty())
{
temp->next = q.top(); // 将元素值最小的首结点插入到新链表后面
temp = temp->next; // 指针后移
q.pop(); // 首结点元素值最小的链表出队
if (temp->next) // 把删除首结点后的链表放入队列中
q.push(temp->next);
}
temp = head;
head = head->next;
delete(temp); // 删除哨兵结点
return head;
}
};
获取更多精彩,请关注「seniusen」!

LeetCode 23 ——合并 K 个排序链表的更多相关文章
- LeetCode 23. 合并K个排序链表(Merge Two Sorted Lists)
23. 合并K个排序链表 23. Merge k Sorted Lists 题目描述 合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. LeetCode23. Merge k S ...
- Java实现 LeetCode 23 合并K个排序链表
23. 合并K个排序链表 合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. 示例: 输入: [ 1->4->5, 1->3->4, 2->6 ] 输 ...
- [LeetCode]23. 合并K个排序链表(优先队列;分治待做)
题目 合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. 示例: 输入: [ 1->4->5, 1->3->4, 2->6 ] 输出: 1 ...
- [LeetCode] 23. 合并K个排序链表
题目链接: https://leetcode-cn.com/problems/merge-k-sorted-lists/ 题目描述: 合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂 ...
- leetcode 23. 合并K个排序链表 JAVA
题目: 合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. 示例: 输入: [ 1->4->5, 1->3->4, 2->6 ] 输出: ...
- LeetCode 23. 合并K个排序链表(Merge k Sorted Lists)
题目描述 合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. 示例: 输入: [ 1->4->5, 1->3->4, 2->6 ] 输出: ...
- 【LeetCode】23.合并K个排序链表
题目描述 23.合并K个排序链表 合并k个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. 示例: 输入: [ 1->4->5, 1->3->4, 2->6 ] ...
- LeetCode题解-23 合并K个排序链表 Hard
合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. 示例: 输入: [ 1->4->5, 1->3->4, 2->6 ] 输出: 1->1-&g ...
- Leetcode题库——23.合并k个排序链表
@author: ZZQ @software: PyCharm @file: mergeKLists.py @time: 2018/10/12 19:55 说明:合并 k 个排序链表,返回合并后的排序 ...
随机推荐
- 用$(this)选择其下带有class的子元素
$(this).find('.son').removeClass("disn")
- 阻止vue事件冒泡的方法
- oracle client安装与配置
(一)安装Oracle client 环境:windows7 64-bit.oracle client 64-bit (1)解压client安装包 (2)双击setup.exe,选择管理员,一直nex ...
- Flask—03-bootstrap与表单
bootstrap与表单 Bootstrap是美国Twitter公司的设计师Mark Otto和Jacob Thornton合作基于HTML.CSS.JavaScript 开发的简洁.直观.强悍的前端 ...
- flask笔记(二)
Flask中的路由 查看整个flask中的路由映射关系 app.url_map from flask import Flask app = Flask(__name__) @app.route(&qu ...
- python2.7打包环境配置
目前python3.x正大行其道,不过有些公司依然使用python2.x,比如说我现在的公司.网上python2.x解决方案还是有些空缺,需要自己去查找. 公司的电脑安装的python2.7,pip也 ...
- Linux centos7 安装python3 及 GCC
1.用wget下载python源码 PYTHON下载 找适合自己的版本,我下载的是3.7.2 2.用tar命令解压下载的文件 tar -zxvf Python-3.7.2.tgz 3.进入目录解压后的 ...
- 吐血分享:QQ群霸屏技术教程2017(效益篇)
懂得如何做群排名了,接下来就要实质性的考虑产出了. 可能,咱们经常发现,一些群里拉人的,进群看某片,5元钱终生,这类是灰色的.其实正规的付费空间也很大. 群利润空间 有工作,有产品,有项目,可以做群排 ...
- Yii 2.0.6 - 从入口到Action执行
defined('YII_DEBUG') or define('YII_DEBUG', true); defined('YII_ENV') or define('YII_ENV', 'dev'); r ...
- ssh 远程命令
远程拷贝文件,scp -r 的常用方法: 1.使用该命令的前提条件要求目标主机已经成功安装openssh-server 如没有安装使用 sudo apt-get install openssh-ser ...