c/c++语言开发共享C++实现LeetCode(148.链表排序)

[leetcode] 148. sort list 链表排序sort a linked list in o(n log n) time using constant sp


[leetcode] 148. sort list 链表排序

sort a linked list in o(n log n) time using constant space complexity.

example 1:

input: 4->2->1->3
output: 1->2->3->4

example 2:

input: -1->5->3->4->0
output: -1->0->3->4->5

常见排序方法有很多,插入排序,选择排序,堆排序,快速排序,冒泡排序,归并排序,桶排序等等。。它们的时间复杂度不尽相同,而这里题目限定了时间必须为o(nlgn),符合要求只有快速排序,归并排序,堆排序,而根据单链表的特点,最适于用归并排序。为啥呢?这是由于链表自身的特点决定的,由于不能通过坐标来直接访问元素,所以快排什么的可能不太容易实现(但是被评论区的大神们打脸,还是可以实现的),堆排序的话,如果让新建结点的话,还是可以考虑的,若只能交换结点,最好还是不要用。而归并排序(又称混合排序)因其可以利用递归来交换数字,天然适合链表这种结构。归并排序的核心是一个 merge() 函数,其主要是合并两个有序链表,这个在 leetcode 中也有单独的题目 merge two sorted lists。由于两个链表是要有序的才能比较容易 merge,那么对于一个无序的链表,如何才能拆分成有序的两个链表呢?我们从简单来想,什么时候两个链表一定都是有序的?就是当两个链表各只有一个结点的时候,一定是有序的。而归并排序的核心其实是分治法 divide and conquer,就是将链表从中间断开,分成两部分,左右两边再分别调用排序的递归函数 sortlist(),得到各自有序的链表后,再进行 merge(),这样整体就是有序的了。因为子链表的递归函数中还是会再次拆成两半,当拆到链表只有一个结点时,无法继续拆分了,而这正好满足了前面所说的“一个结点的时候一定是有序的”,这样就可以进行 merge 了。然后再回溯回去,每次得到的都是有序的链表,然后进行 merge,直到还原整个长度。这里将链表从中间断开的方法,采用的就是快慢指针,大家可能对快慢指针找链表中的环比较熟悉,其实找链表中的中点同样好使,因为快指针每次走两步,慢指针每次走一步,当快指针到达链表末尾时,慢指针正好走到中间位置,参见代码如下:

c++ 解法一:

  class solution {  public:      listnode* sortlist(listnode* head) {          if (!head || !head->next) return head;          listnode *slow = head, *fast = head, *pre = head;          while (fast && fast->next) {              pre = slow;              slow = slow->next;              fast = fast->next->next;          }          pre->next = null;          return merge(sortlist(head), sortlist(slow));      }      listnode* merge(listnode* l1, listnode* l2) {          listnode *dummy = new listnode(-1);          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;      }  };

java 解法一:

  public class solution {      public listnode sortlist(listnode head) {          if (head == null || head.next == null) return head;          listnode slow = head, fast = head, pre = head;          while (fast != null && fast.next != null) {              pre = slow;              slow = slow.next;              fast = fast.next.next;          }          pre.next = null;          return merge(sortlist(head), sortlist(slow));      }      public listnode merge(listnode l1, listnode l2) {          listnode dummy = new listnode(-1);          listnode cur = dummy;          while (l1 != null && l2 != null) {              if (l1.val < l2.val) {                  cur.next = l1;                  l1 = l1.next;              } else {                  cur.next = l2;                  l2 = l2.next;              }              cur = cur.next;          }          if (l1 != null) cur.next = l1;          if (l2 != null) cur.next = l2;          return dummy.next;      }  }

下面这种方法也是归并排序,而且在merge函数中也使用了递归,这样使代码更加简洁啦~

c++ 解法二:

  class solution {  public:      listnode* sortlist(listnode* head) {          if (!head || !head->next) return head;          listnode *slow = head, *fast = head, *pre = head;          while (fast && fast->next) {              pre = slow;              slow = slow->next;              fast = fast->next->next;          }          pre->next = null;          return merge(sortlist(head), sortlist(slow));      }      listnode* merge(listnode* l1, listnode* l2) {          if (!l1) return l2;          if (!l2) return l1;          if (l1->val < l2->val) {              l1->next = merge(l1->next, l2);              return l1;          } else {              l2->next = merge(l1, l2->next);              return l2;          }      }  };

java 解法二:

  public class solution {      public listnode sortlist(listnode head) {          if (head == null || head.next == null) return head;          listnode slow = head, fast = head, pre = head;          while (fast != null && fast.next != null) {              pre = slow;              slow = slow.next;              fast = fast.next.next;          }          pre.next = null;          return merge(sortlist(head), sortlist(slow));      }      public listnode merge(listnode l1, listnode l2) {          if (l1 == null) return l2;          if (l2 == null) return l1;          if (l1.val < l2.val) {              l1.next = merge(l1.next, l2);              return l1;          } else {              l2.next = merge(l1, l2.next);              return l2;          }      }  }

github 同步地址:

类似题目:

merge two sorted lists

sort colors

insertion sort list

参考资料:

到此这篇关于c++实现leetcode(148.链表排序)的文章就介绍到这了,更多相关c++实现链表排序内容请搜索<计算机技术网(www.ctvol.com)!!>以前的文章或继续浏览下面的相关文章希望大家以后多多支持<计算机技术网(www.ctvol.com)!!>!

需要了解更多c/c++开发分享C++实现LeetCode(148.链表排序),都可以关注C/C++技术分享栏目—计算机技术网(www.ctvol.com)!

本文来自网络收集,不代表计算机技术网立场,如涉及侵权请联系管理员删除。

ctvol管理联系方式QQ:251552304

本文章地址:https://www.ctvol.com/c-cdevelopment/677445.html

(0)
上一篇 2021年7月28日
下一篇 2021年7月28日

精彩推荐