23.合并k个有序链表
合并 k 个排序链表,返回合并后的排序链表。请分析和描述算法的复杂度。
示例:
输入:
[
1->4->5,
1->3->4,
2->6
]
输出: 1->1->2->3->4->4->5->6
我们熟悉的是两个链表之间的合并,这道题的难点在于k值是不固定的,并不知道要合并多少链表。一种很常见的想法就是使用分治法,因为合并两个有序链表是合并k个有序链表的子问题。
分治法一
多次进行两个链表之间的合并,将k个有序链表转化为(k+1)/2个有序链表。例如有6个链表时,想要合并为3个链表,则链表0与链表3合并得到新的链表0,链表1与链表4合并得到新的链表1,链表2与链表5合并得到新的链表2。最终所有链表结点都合并到了链表0,即为所求结果。
举例说明该过程:
[[1,4,5],[1,3,4],[2,6]]->[[1,2,4,5,6],[1,3,4]]->[[1,1,2,3,4,4,5,6]]
ListNode*mergeTwoLists(ListNode*l1,ListNode*l2){
ListNode*dummy=new ListNode(),*cur=dummy;
while(l1&&l2){
if(l1->val<l2->val){
cur->next=l1;
l1=l1->next;
}
else{
cur->next=l2;
l2=l2->next;
}
cur=cur->next;
}
if(l1) cur->next=l1;
if(l2) cur->next=l2;
return dummy->next;
}
ListNode* mergeKLists(vector<ListNode*>& lists) {
int n=lists.size();
//n=0返回空,n=1返回lists[0]
if(n==0) return NULL;
while(n>1){
//k为合并的步长
int k=(n+1)/2;
//如果n为奇数则正中间链表不参与合并
for(int i=0;i<n/2;++i){
//将lists[i]和lists[i+k]合并到lists[i]
lists[i]=mergeTwoLists(lists[i],lists[i+k]);
}
//将n个链表经过两两合并之后转为(n+1)/2个链表
n=k;
}
//n=1退出循环,此时所有链表都合并进了lists[0]
return lists[0];
}
分治法二
类似归并排序
ListNode*mergeTwoLists(ListNode*l1,ListNode*l2){
ListNode*dummy=new ListNode(),*cur=dummy;
while(l1&&l2){
if(l1->val<l2->val){
cur->next=l1;
l1=l1->next;
}
else{
cur->next=l2;
l2=l2->next;
}
cur=cur->next;
}
if(l1) cur->next=l1;
if(l2) cur->next=l2;
return dummy->next;
}
ListNode*helper(vector<ListNode*>&lists,int left,int right){
if(left==right) return lists[left];
int mid=(left+right)/2;
ListNode*l1=helper(lists,left,mid);
ListNode*l2=helper(lists,mid+1,right);
//当前递归层的返回值
return mergeTwoLists(l1,l2);
}
ListNode* mergeKLists(vector<ListNode*>& lists) {
int n=lists.size();
if(n==0) return NULL;
return helper(lists,0,n-1);
}
最小堆
首先将k个链表的首元素添加到最小堆,取出最小元素加入到结果链表中。再将所加元素的下一个元素加入到最小堆(此时的最小元素要么还在最小堆中,要么是之前所加元素的下一个元素),取出最小元素加入到结果链表中。以此类推,直到堆中没有元素为止。
public ListNode mergeKLists(ListNode[] lists) {
int n=lists.length;
PriorityQueue<ListNode> pq=new PriorityQueue<ListNode>(new Comparator<ListNode>(){
public int compare(ListNode l1,ListNode l2){
return l1.val-l2.val;
}
});
ListNode dummy=new ListNode(-1);
ListNode cur=dummy;
for(int i=0;i<n;++i) {
if(lists[i]!=null){
pq.add(lists[i]);
lists[i]=lists[i].next;
}
}
while(!pq.isEmpty()){
cur.next=pq.poll();
cur=cur.next;
//将取出元素的下一个元素加入堆中
if(cur.next!=null) pq.add(cur.next);
}
return dummy.next;
}
计数排序
依次遍历所有链表,记录所有元素中的最小值和最大值,那么每个元素值必定在它们两者之间。同时记录所有元素出现的次数。然后就可以从最小值遍历到最大值,根据其出现的次数建立相应个数的结点。
public ListNode mergeKLists(ListNode[] lists) {
int n=lists.length;
ListNode dummy=new ListNode(-1),cur=dummy;
HashMap<Integer,Integer> hashmap=new HashMap<>();
int min=Integer.MAX_VALUE;
int max=Integer.MIN_VALUE;
for(int i=0;i<n;++i){
while(lists[i]!=null){
if(hashmap.containsKey(lists[i].val))
hashmap.put(lists[i].val,hashmap.get(lists[i].val)+1);
else hashmap.put(lists[i].val,1);
if(lists[i].val<min) min=lists[i].val;
if(lists[i].val>max) max=lists[i].val;
lists[i]=lists[i].next;
}
}
for(int i=min;i<=max;++i){
if(!hashmap.containsKey(i)) continue;
for(int j=0;j<hashmap.get(i);++j){
cur.next=new ListNode(i);
cur=cur.next;
}
}
return dummy.next;
}
leetcode原题:https://leetcode-cn.com/problems/merge-k-sorted-lists
23.合并k个有序链表的更多相关文章
- [LeetCode] 23. Merge k Sorted Lists 合并k个有序链表
Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity. E ...
- [LeetCode] Merge k Sorted Lists 合并k个有序链表
Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity. 这 ...
- LeetCode 23. 合并K个排序链表(Merge Two Sorted Lists)
23. 合并K个排序链表 23. Merge k Sorted Lists 题目描述 合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. LeetCode23. Merge k S ...
- leetcode python 012 hard 合并k个有序链表
#[LeetCode] Merge k Sorted Lists 合并k个有序链表(升序) import numpy as npimport time class Node(object): d ...
- 023 Merge k Sorted Lists 合并K个有序链表
合并K个有序链表,并且作为一个有序链表的形式返回.分析并描述它的复杂度. 详见:https://leetcode.com/problems/merge-k-sorted-lists/descripti ...
- 【LeetCode】23.合并K个排序链表
题目描述 23.合并K个排序链表 合并k个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. 示例: 输入: [ 1->4->5, 1->3->4, 2->6 ] ...
- Java实现 LeetCode 23 合并K个排序链表
23. 合并K个排序链表 合并 k 个排序链表,返回合并后的排序链表.请分析和描述算法的复杂度. 示例: 输入: [ 1->4->5, 1->3->4, 2->6 ] 输 ...
- [LeetCode题解]23. 合并K个升序链表 | 分治 + 递归
方法一:分治 + 递归 解题思路 在21. 合并两个有序链表,我们知道如何合并两个有序链表.而本题是合并 k 个有序链表,可以通过大问题拆分成小问题解决,即把 k 个链表,拆分成 k/2 个链表组,俩 ...
- LeetCode 23 ——合并 K 个排序链表
1. 题目 2. 解答 2.1. 方法一 在 合并两个有序链表 的基础上,我们很容易想到第一种解法,首先我们将第一个链表和第二个链表合并成一个新的链表,然后再往后依次合并接下来的每个链表即可. 假设每 ...
随机推荐
- Kubernetes环境Traefik部署与应用
本作品由Galen Suen采用知识共享署名-非商业性使用-禁止演绎 4.0 国际许可协议进行许可.由原作者转载自个人站点. 概述 本文用于整理基于Kubernetes环境的Traefik部署与应用, ...
- 2021.9.12周六PAT甲级考试复盘与总结
周六PAT甲级考试复盘与总结 先说结论:仍未步入"高手"行列:现在的学习节奏与方法是对的,有十万分的必要坚持下去. 题目 知识点 分数 T1 前缀和.二分 11 / 20 T2 排 ...
- Linux内核学习之2号进程kthreadd
Author : Toney Email : vip_13031075266@163.com Date : 2020.12.04 Copyright : ...
- python3 爬虫五大模块之一:爬虫调度器
Python的爬虫框架主要可以分为以下五个部分: 爬虫调度器:用于各个模块之间的通信,可以理解为爬虫的入口与核心(main函数),爬虫的执行策略在此模块进行定义: URL管理器:负责URL的管理,包括 ...
- mybatis和hibernate区别
一.本质区别和应用场景
- 以人为本打造“超职季”IP,58同城精准匹配企业招聘与打工人
撰文 |懂懂 编辑 | 秦言 来源:懂懂笔记 在大手笔培育IP的背后,58同城是如何考量的? 在餐厅当服务员的李阿姨今年54岁了.她的女儿马上研究生毕业,非常喜欢陈伟霆,手机屏保都是他.李阿姨没想到, ...
- 生产环境部署高可用Rancher
环境准备: IP hostname role 192.168.200.150 nginx LB 192.168.200.151 master01-151 docker-ce/rke/helm/kube ...
- 在树莓派用C#+Winform实现传感器监测
最近学校里发了个任务,说要做一个科技节小发明,然后我就掏出我的树莓派准备大干一场. 调料 Raspberry Pi 3B+ 树莓派GPIO扩展板 3.5寸电容触摸屏(GPIO接口) 土壤湿度传感器(G ...
- Spring Cloud Hystrix 学习(一)
在学习Hystrix之前,首先引入一个问题场景,服务雪崩.如下图所示: 可以看到,三个入口服务A.B.C最终都会请求到服务T.当服务T的请求过载,打满CPU都无法匹配请求的频率时,同步调用的上级服务就 ...
- mogoose 创建数据库并增删改查
下载 npm i mongoose -s 连接数据库 const mongoose = require("mongoose"); mongoose.connect(dbURL); ...