Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity.

思路I: 选择排序

每次都比较各个list的头指针所指的val,取最小的那个。时间复杂度O(n*k)

class Solution {
public:
ListNode *mergeKLists(vector &lists) {
if(lists.empty()) return NULL;
ListNode *head = new ListNode(INT_MIN); //创建头指针,不指向任何List元素;最后返回的是head->next
head->next = lists[];
ListNode *p1;
ListNode *p2;
ListNode *tmp;
for(int i = ; i< lists.size(); i++)
{
p1 = head;
p2 = lists[i];
while(p1->next && p2)
{
if(p1->next->val <= p2->val) p1 = p1->next;
else
{
tmp = p1->next;
p1->next = p2;
p1 = p1->next;
p2 = p2->next;
p1->next = tmp;
}
}
if(p2)
{
p1->next = p2;
}
}
return head->next;
}
};

Result:Time Limit Exceeded

思路II: 最小堆。

时间复杂度:堆排序其实也是一种选择排序。只不过直接选择排序中,为了从R[1...n]中选择最大记录,需比较n-1次,然后从R[1...n-2]中选择最大记录需比较n-2次。事实上这n-2次比较中有很多已经在前面的n-1次比较中已经做过,而树形选择排序恰好利用树形的特点保存了部分前面的比较结果,因此可以减少比较次数。时间复杂度为O(nlogk)

堆排序思想参考:http://jingyan.baidu.com/article/5225f26b057d5de6fa0908f3.html

class Solution {
public:
ListNode *mergeKLists(vector<ListNode *> &lists) {
// 使用堆排序,
// 1. 选出每个链表的头来插入小顶堆中,
// 2. 再把堆顶接入合并链表中,
// 3. 被选出的指针后移再加入小顶堆中,回到2
// 4. 最后所有链表都为空时,返回合并链表的头指针
if(lists.empty()) return nullptr;
vector<ListNode* > heap;
heap.push_back(); //padding
// 1. 选出每个链表的头来插入小顶堆中,
for(int i = ; i != lists.size(); i ++){
if(lists[i]) heap.push_back(lists[i]);
}
makeHeap(heap);
// 2. 再把堆顶接入合并链表中,
ListNode head(-); // 合并链表的表头
ListNode* p = &head;
while(heap.size()>){
auto minNode = heap[];
p->next = minNode; // 接入链表
p = p->next;
// 3. 被选出的指针后移再加入小顶堆中,回到2
auto next = minNode->next;
if(next) {
heap[] = next;
}else{
swap(heap[], heap[heap.size()-]);
heap.pop_back();
}
minHeap(heap, );//加入新元素到堆顶后,自上向下调整
}
// 4. 最后所有链表都为空时,返回合并链表的头指针
return head.next;
}
// 建立小顶堆
// 自底向上
void makeHeap(vector<ListNode*> &heap){
// 从最后一个元素的父节点开始建立小顶堆
for(int i = (heap.size()-)/; i > ; i --){
minHeap(heap, i);
}
}
// 小顶堆,以第i个元素为根建立小顶堆
//位置从1开始,取元素时记得-1
// 自顶向下
void minHeap(vector<ListNode*> &heap, int i){
int l = i*;
int r = l+;
int least(i);
// 算出最小元素的位置
if((l< heap.size()) && heap[l]->val<heap[i]->val ){
// 如果没有超过边界并且左孩子比父亲小,则换
least = l;
}
if(r<heap.size() && heap[r]->val<heap[least]->val){
// 如果没有超过边界并且右孩子最小,则换
least = r;
}
if(least != i){
swap(heap[i], heap[least]);
minHeap(heap, least);//换了之后,继续向下调整
}
}
};

最小堆在C++中可以用优先队列来实现

/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/ struct cmp {
bool operator () (ListNode *a, ListNode *b) {
return a->val > b->val;
}
}; class Solution {
public:
ListNode* mergeKLists(vector<ListNode*>& lists) {
priority_queue<ListNode*,vector<ListNode*>,cmp> queue;
for (int i = ; i < lists.size(); i++) {
if (lists[i] != NULL) {
queue.push(lists[i]);
}
} ListNode *head = new ListNode(INT_MIN);
ListNode *prev = head, *temp;
while (!queue.empty()) {
temp = queue.top();
queue.pop();
prev->next = temp;
prev = prev->next; if (temp->next != NULL) {
queue.push(temp->next);
}
}
return head->next;
}
};

优先队列默认从大到小排序,即大顶堆,所以这里我们需要改成小顶堆:

1. 基本类型(例如int)都可以用greater<type>声明小顶堆优先队列:

priority_queue<int, vector<int>, greater<int>> q

2. 自定义的数据结构必须自定义比较函数(如以上代码中定义的cmp),或者自定义 > 操作:

bool operator > (ListNode* a, ListNode* b){
return a->val > b->val;
}

思路III:归并排序

初始状态:6,202,100,301,38,8,1
第一次归并后:{6,202},{100,301},{8,38},{1},比较次数:3;
第二次归并后:{6,100,202,301},{1,8,38},比较次数:4;
第三次归并后:{1,6,8,38,100,202,301},比较次数:4;
reference: http://baike.baidu.com/link?url=ayX3MQx_CrmcjOxkL7EKhXukLH9pJKJsD1XDMaP6eQwvFfc-BtnQBUTsElRafXbxqhCFOIlKC5VsL14LgjEjIK 
归并排序时间复杂度O(nlogn)
/**
* 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) {
if(lists.empty()) return NULL; mergeSort(lists, , lists.size()-);
return lists[];
}
void mergeSort(vector<ListNode*>& lists, int start, int end){
if(start==end) return;
int mid = (start + end) >> ;
mergeSort(lists, start, mid);
mergeSort(lists, mid+, end);
merge(lists,start, mid+);
} void merge(vector<ListNode*>& lists, int lst1, int lst2){
ListNode* root = NULL;
ListNode* current = NULL; while(lists[lst1] && lists[lst2]){
if(lists[lst1]->val <= lists[lst2]->val){
if(!root){
root = lists[lst1];
current = root;
}
else{
current->next = lists[lst1];
current = current->next;
}
lists[lst1] = lists[lst1]->next;
}
else{
if(!root){
root = lists[lst2];
current = root;
}
else{
current->next = lists[lst2];
current = current->next;
}
lists[lst2] = lists[lst2]->next;
}
} while(lists[lst1]){
if(!root){
root = lists[lst1];
current = root;
}
else{
current->next = lists[lst1];
current = current->next;
}
lists[lst1] = lists[lst1]->next;
}
while(lists[lst2]){
if(!root){
root = lists[lst2];
current = root;
}
else{
current->next = lists[lst2];
current = current->next;
}
lists[lst2] = lists[lst2]->next;
} lists[lst1] = root;
}
};

23.Merge k Sorted Lists (Array, Queue; Sort)的更多相关文章

  1. 刷题23. Merge k Sorted Lists

    一.题目说明 这个题目是23. Merge k Sorted Lists,归并k个有序列表生成一个列表.难度为Hard,实际上并不难,我一次提交就对了. 二.我的解答 就是k路归并,思路很简单,实现也 ...

  2. [Leetcode][Python]23: Merge k Sorted Lists

    # -*- coding: utf8 -*-'''__author__ = 'dabay.wang@gmail.com' 23: Merge k Sorted Listshttps://oj.leet ...

  3. 21.Merge Two Sorted Lists 、23. Merge k Sorted Lists

    21.Merge Two Sorted Lists 初始化一个指针作为开头,然后返回这个指针的next class Solution { public: ListNode* mergeTwoLists ...

  4. 23. Merge k Sorted Lists - LeetCode

    Question 23. Merge k Sorted Lists Solution 题目大意:合并链表数组(每个链表中的元素是有序的),要求合并后的链表也是有序的 思路:遍历链表数组,每次取最小节点 ...

  5. 蜗牛慢慢爬 LeetCode 23. Merge k Sorted Lists [Difficulty: Hard]

    题目 Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity ...

  6. [LeetCode] 23. Merge k Sorted Lists ☆☆☆☆☆

    转载:https://leetcode.windliang.cc/leetCode-23-Merge-k-Sorted-Lists.html 描述 Merge k sorted linked list ...

  7. [LeetCode] 23. Merge k Sorted Lists ☆☆

    Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity. 解 ...

  8. [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 ...

  9. [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 ...

随机推荐

  1. Total Defense Anti-Virus – 免费6个月

    Total Defense Anti-Virus 具有病毒和间谍软件保护,免费的专家远程安装和设置,快速扫描引擎不会拖慢电脑速度,基于云保护.高级Rootkit保护,自动扫描驱动器和内存等等.官方网站 ...

  2. 免费一年MAP2014+6个月免费MIS2014

    活动页面:http://promos.mcafee.com/en-us/Affiliates/400/Lp/np2815_mavp.aspx?rfhs=1&LPName=np2815_mavp ...

  3. java静态代理和动态代理(一)

    代理Proxy: Proxy代理模式是一种结构型设计模式,主要解决的问题是:在直接访问对象时带来的问题. 代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问.代理类负责为 ...

  4. LightOJ - 1396 :Palindromic Numbers (III)(逐位确定法)

    Vinci is a little boy and is very creative. One day his teacher asked him to write all the Palindrom ...

  5. 中南林业科技大学第十一届程序设计大赛- I:背包问题

    链接:https://www.nowcoder.com/acm/contest/124/I来源:牛客网 时间限制:C/C++ 2秒,其他语言4秒空间限制:C/C++ 131072K,其他语言26214 ...

  6. bug 报告

    准备工作:首先我下载了Git Bash这个软件,利用了老师所给的git命令,批量的pull所有同学第二次作业的代码.截图如下: 一.被测试程序的作者:方铭 程序git地址:https://coding ...

  7. [Luogu4899][IOI2018] werewolf 狼人

    luogu sol \(\mbox{IOI2018}\)的出题人有没有看过\(\mbox{NOI2018}\)的题目呀... \(\mbox{Kruskal}\)重构树+二维数点. 题目相当于是问你从 ...

  8. CH1807 Necklace

    题意 背景 有一天,袁☆同学绵了一条价值连城宝石项链,但是,一个严重的问题是,他竟然忘记了项链的主人是谁!在得知此事后,很多人向☆同学发来了很多邮件,都说项链是自己的,要求他归还(显然其中最多只有一个 ...

  9. python try except, 异常处理

    http://www.runoob.com/python/python-exceptions.html http://blog.sciencenet.cn/blog-3031432-1059523.h ...

  10. poj3311 经典tsp问题

    题目的大概意思就是一个人到一些城市送披萨,要求找到一条路径可以遍历每个城市后返回出发点,而且路径距离最短.最后输出最短距离就可以. 注意:每个城市可反复訪问多次. 因为题中明白说了两个城市间的直接可达 ...