题目描述(中等难度)

给一个链表,然后依次头尾头尾头尾取元素,组成新的链表。

解法一 存储

链表的缺点就是不能随机存储,当我们想取末尾元素的时候,只能从头遍历一遍,很耗费时间。第二次取末尾元素的时候,又得遍历一遍。

所以先来个简单粗暴的想法,把链表存储到线性表中,然后用双指针依次从头尾取元素即可。

解法二 递归

参考 。

解法一中也说到了,我们的问题就是取尾元素的时候,需要遍历一遍链表。

如果我们的递归函数能够返回当前头元素对应的尾元素,并且将头元素和尾元素之间的链表按要求完成,那就变得简单了。

143. Reorder List - 图1

然后我们把之前的 tail.next 返回就是外层 head 对应的 tail 了。

递归出口的话,如果只有一个节点,那么我们只需要将 head.next 返回。

  1. if (len == 1) {
  2. ListNode outTail = head.next;
  3. head.next = null;
  4. return outTail;
  5. }

如果是两个节点,我们需要将 head.next.next 返回。

然后总体的代码就是下边的样子

  1. public void reorderList(ListNode head) {
  2. if (head == null || head.next == null || head.next.next == null) {
  3. return;
  4. }
  5. int len = 0;
  6. ListNode h = head;
  7. //求出节点数
  8. while (h != null) {
  9. len++;
  10. h = h.next;
  11. }
  12. reorderListHelper(head, len);
  13. }
  14. private ListNode reorderListHelper(ListNode head, int len) {
  15. if (len == 1) {
  16. ListNode outTail = head.next;
  17. }
  18. if (len == 2) {
  19. ListNode outTail = head.next.next;
  20. head.next.next = null;
  21. return outTail;
  22. }
  23. //得到对应的尾节点,并且将头结点和尾节点之间的链表通过递归处理
  24. ListNode tail = reorderListHelper(head.next, len - 2);
  25. ListNode subHead = head.next;//中间链表的头结点
  26. head.next = tail;
  27. ListNode outTail = tail.next; //上一层 head 对应的 tail
  28. tail.next = subHead;
  29. return outTail;
  30. }

解法三

参考 这里,主要是利用到一头一尾取元素的特性。

主要是三步,举个例子。

第二步链表逆序的话,在 讨论过了,有迭代和递归的两种方式,迭代的话主要利用两个指针,依次逆转。

第三步的话就很简单了,两个指针分别向后移动就可以。

  1. public void reorderList(ListNode head) {
  2. if (head == null || head.next == null || head.next.next == null) {
  3. return;
  4. }
  5. //找中点,链表分成两个
  6. ListNode slow = head;
  7. ListNode fast = head;
  8. while (fast.next != null && fast.next.next != null) {
  9. slow = slow.next;
  10. fast = fast.next.next;
  11. }
  12. ListNode newHead = slow.next;
  13. slow.next = null;
  14. newHead = reverseList(newHead);
  15. //链表节点依次连接
  16. while (newHead != null) {
  17. ListNode temp = newHead.next;
  18. newHead.next = head.next;
  19. head.next = newHead;
  20. head = newHead.next;
  21. newHead = temp;
  22. }
  23. }
  24. private ListNode reverseList(ListNode head) {
  25. if (head == null) {
  26. return null;
  27. }
  28. ListNode tail = head;
  29. head = head.next;
  30. tail.next = null;
  31. while (head != null) {
  32. ListNode temp = head.next;
  33. head.next = tail;
  34. tail = head;
  35. head = temp;
  36. }
  37. }

解法一利用空间去存储就很简单了,解法二递归的思想也很经典,自己也想了很久,看到作者的思路才恍然大悟,判断当前 定义递归出口很巧妙。解法三主要就是对题目的理解,关键就是利用一头一尾取元素的特性。

添加好友一起进步~

如果觉得有帮助的话,可以点击 这里 给一个 star 哦 ^^

如果想系统的学习数据结构和算法,强烈推荐一个我之前学过的课程,可以点击 查看详情