Sort List题解

题目来源:https://leetcode.com/problems/sort-list/description/


Description

Sort a linked list in O(n log n) time using constant space complexity.

Solution

/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/ class Solution {
public:
int partition(vector<int>& arr, int start, int end) {
int pivot = arr[start];
int low = start, high = end;
while (low < high) {
while (arr[low] <= pivot && low < high)
++low;
while (arr[high] > pivot && low < high)
--high;
swap(arr[low], arr[high]);
}
if (arr[low] > pivot)
--low;
arr[start] = arr[low];
arr[low] = pivot;
return low;
} void quickSort(vector<int>& arr, int start, int end) {
if (start >= end)
return;
int pivotIndex = partition(arr, start, end);
quickSort(arr, start, pivotIndex - 1);
quickSort(arr, pivotIndex + 1, end);
} ListNode* sortList(ListNode* head) {
if (head == NULL || head -> next == NULL) {
return head;
} ListNode* temp = head;
vector<int> arr; while (temp != NULL) {
arr.push_back(temp -> val);
temp = temp -> next;
} quickSort(arr, 0, arr.size() - 1); temp = head;
int i = 0;
while (temp != NULL) {
temp -> val = arr[i++];
temp = temp -> next;
} return head;
}
};

解题描述

这道题考察的是链表排序。我首先想到的是把链表里面的元素拷贝到一个vector里面然后再快排,这就跟平时对数组快排的操作一样了,然后写起来也比较好写。但是AC之后觉得,2次拷贝元素的时间会不会有点多?而且还带来了额外的空间开销。于是自己还是写多了一个针对链表的快排:

class Solution {
public:
void swap(int& a, int& b) {
int t = a;
a = b;
b = t;
} ListNode* partition(ListNode* low, ListNode* high) {
int key = low -> val;
ListNode *locNode = low; // the location node the that key will locate at last
for (ListNode* tempNode = low -> next; tempNode != high; tempNode = tempNode -> next) {
if (tempNode -> val < key) {
locNode = locNode -> next;
swap(locNode -> val, tempNode -> val);
}
}
swap(low -> val, locNode -> val);
return locNode;
} void quickSort(ListNode* head, ListNode* tail) {
ListNode* posNode;
if (head != tail && head -> next != tail) { // left close, right open interval
posNode = partition(head, tail);
quickSort(head, posNode);
quickSort(posNode -> next, tail);
}
return;
} ListNode* sortList(ListNode* head) {
quickSort(head, NULL);
return head;
}
};

但是实际跑出来结果却是(后提交的是用链表快排):



可能因为vector底层是用数组实现的,访问速度会比链表快一些,所以就算加上了拷贝元素的时间,总体的时间复杂度还是要低于直接对链表进行快排。

更优解法

2018.2.3更新

对链表来说,选择归并排序才是更明智的选择

  • 链表无法像数组一样快速随机访问元素,要求排序算法元素访问次数较少且稳定
  • 使用归并排序处理链表不需要像数组一样使用额外的内存,合并链表的时候只需要进行指针连接即可

下面给出链表递归归并排序的实现:

class Solution {
private:
ListNode* merge(ListNode* head1, ListNode* head2) {
ListNode tempHead(0);
ListNode *curNode = &tempHead;
while (head1 && head2) {
if (head1 -> val <= head2 -> val) {
curNode -> next = head1;
head1 = head1 -> next;
} else {
curNode -> next = head2;
head2 = head2 -> next;
}
curNode = curNode -> next;
}
while (head1) {
curNode -> next = head1;
head1 = head1 -> next;
curNode = curNode -> next;
}
while (head2) {
curNode -> next = head2;
head2 = head2 -> next;
curNode = curNode -> next;
}
return tempHead.next;
}
public:
ListNode* sortList(ListNode* head) {
if (!head)
return NULL;
if (head -> next == NULL)
return head;
if (head -> next -> next == NULL) {
if (head -> val <= head -> next -> val) {
return head;
} else {
ListNode *newHead = head -> next;
newHead -> next = head;
head -> next = NULL;
return newHead;
}
}
ListNode *mid = head, *tail = head;
while (tail && tail -> next) {
mid = mid -> next;
tail = tail -> next -> next;
}
ListNode* head2 = sortList(mid -> next);
mid -> next = NULL;
head = sortList(head);
return merge(head, head2);
}
};

[Leetcode Week2]Sort List的更多相关文章

  1. [Leetcode Week2]Sort Colors

    Sort Colors题解 原创文章,拒绝转载 题目来源:https://leetcode.com/problems/sort-colors/description/ Description Give ...

  2. C#版 - LeetCode 148. Sort List 解题报告(归并排序小结)

    leetcode 148. Sort List 提交网址: https://leetcode.com/problems/sort-list/  Total Accepted: 68702 Total ...

  3. 待字闺中之快排单向链表;leetcode之Sort List

    题目来源.待字闺中.原创@陈利人 .欢迎大家继续关注微信公众账号"待字闺中" 分析:思路和数据的高速排序一样,都须要找到一个pivot元素.或者节点. 然后将数组或者单向链表划分为 ...

  4. LeetCode——Insertion Sort List

    LeetCode--Insertion Sort List Question Sort a linked list using insertion sort. Solution 我的解法,假设第一个节 ...

  5. [LeetCode] Wiggle Sort II 摆动排序

    Given an unsorted array nums, reorder it such that nums[0] < nums[1] > nums[2] < nums[3]... ...

  6. [LeetCode] Wiggle Sort 摆动排序

    Given an unsorted array nums, reorder it in-place such that nums[0] <= nums[1] >= nums[2] < ...

  7. [LeetCode] Insertion Sort List 链表插入排序

    Sort a linked list using insertion sort. 链表的插入排序实现原理很简单,就是一个元素一个元素的从原链表中取出来,然后按顺序插入到新链表中,时间复杂度为O(n2) ...

  8. 【Leetcode】Sort List JAVA实现

    Sort a linked list in O(n log n) time using constant space complexity. 1.分析 该题主要考查了链接上的合并排序算法. 2.正确代 ...

  9. [LeetCode] 148. Sort List 解题思路

    Sort a linked list in O(n log n) time using constant space complexity. 问题:对一个单列表排序,要求时间复杂度为 O(n*logn ...

随机推荐

  1. 虚拟现实-VR-UE4-创建C++版工程

    首先,创建C++版本的UE4 项目工程,我使用的是4.12.3版本,据了解,新版本后面的编译都是vs2015 所以,想要创建C++版本的工程,就需要安装vs2015 至于vs2015的安装,自己百度吧 ...

  2. 数据库学习(三) sql语句中添加函数 to_char,round,连接符||

    ** to char 是把日期或数字转换为字符串  to date 是把字符串转换为数据库中得日期类型  参考资料:https://www.cnblogs.com/hllnj2008/p/533296 ...

  3. url解读

    我刚刚学习的时候,我抓到包不知道哪个是协议.哪个是是服务器地址.哪个是端口号...不知道有没有老铁遇到跟我一样的. 接口:http://172.168.12.0:8888/old/login.do 解 ...

  4. ajax中用jsonp接收json数据

    最近在做查快递网页时遇到一个问题,调用的快递100的api,但是快递100api不允许跨域请求,就是用127.0.0.1发的请求会直接被拦截. 只是个简单的网页,不想自己做服务器转发,最后找到了一个y ...

  5. 算法(11)Find All Duplicates in an Array

    题目:数组的长度是n,里面的数是1到n,其中肯定有重复的,找到里面重复的数字 思路:自己想愣是没有想出来,直接看答案,关键点是看nums[i]和nums[nums[i]-1]之间的关系,遍历整个数组, ...

  6. hdu 1284 钱币兑换问题 (递推 || DP || 母函数)

    钱币兑换问题 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Subm ...

  7. 【题解】MUTC2013idiots

    我是先知道的这题是FFT然后再做的,知道是FFT其实就是个套路题啦.首先,我们容易发现 \(P = \frac{a}{b}\) 其中a表示合法的方案数,而b表示全部的方案数. b的值即为\(C\lef ...

  8. 【题解】HEOI2013Eden 的新背包问题

    这题真的神奇了……蜜汁复杂度(`・ω・´) 应该是一个比较连贯的思维方式:去掉一个物品,那么我们转移的时候不考虑它就好了呗.考虑暴力:每一次都对剩余的n - 1个物品进行多重背包转移,获得答案.既然可 ...

  9. 【ZJ选讲·BZOJ 5073】

    小A的咒语 给出两个字符串A,B (len<=105) 现在可以把A串拆为任意段,然后取出不超过 x 段,按在A串中的前后顺序拼接起来 问是否可以拼出B串. [题解]       ①如果遇 ...

  10. [COGS 622] [NOIP2011] 玛雅游戏 模拟

    整个模拟的关键除了打出来就是一个剪枝:对于两个左右相邻的块你不用再走←,因为走→是等效的 #include<cstdio> #include<cstring> #include ...