⭐算法入门⭐《链表》中等02 —— LeetCode 143. 重排链表

Posted 英雄哪里出来

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了⭐算法入门⭐《链表》中等02 —— LeetCode 143. 重排链表相关的知识,希望对你有一定的参考价值。

一、题目

1、题目描述

  给定一个单链表 L L L 的头节点 h e a d head head ,单链表 L L L 表示为: L 0 → L 1 → . . . → L n − 1 → L n L_0 \\to L_1 \\to ... \\to L_{n-1} \\to L_n L0L1...Ln1Ln请将其重新排列后变为: L 0 → L n → L 1 → L n − 1 → L 2 → L n − 2 → . . . L_0 \\to L_n \\to L1 \\to L_{n-1} \\to L_2 \\to L_{n-2} \\to ... L0LnL1Ln1L2Ln2...不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。
  样例输入: head = [1,2,3,4,5,6,7]
  样例输出: [1,7,2,6,3,5,4]

2、基础框架

  • C语言 版本给出的基础框架代码如下:
void reorderList(struct ListNode* head) {

}

3、原题链接

LeetCode 143. 重排链表
剑指 Offer II 026. 重排链表

二、解题报告

1、思路分析

  1)首先,利用快慢指针,将链表拆成两个链表,第一个链表为前半部分,第二个链表为后半部分;
  2)然后,将第二个链表进行翻转;
  3)最后,将两个链表进行合并。

2、时间复杂度

  三步操作都是 O ( n ) O(n) O(n) 的,并且是互相独立的部分。

3、代码详解

// 奇数:  L1 -> L2 -> L3  返回  L2
// 偶数:  L1 -> L2        返回  L2
struct ListNode* getHalf(struct ListNode* head) {
    struct ListNode *prev, *slow, *fast, *half;
    if(head == NULL) {
        return head;
    }
    prev = NULL;
    slow = head;
    fast = head;
    while(fast && fast->next) {
        prev = slow;
        slow = slow->next;
        fast = fast->next->next;
    }
    half = slow;
    if(prev)
        prev->next = NULL;
    return half;
}

struct ListNode *removeNextAndReturn(struct ListNode* now) {   // (1)
    struct ListNode *retNode;
    if(now == NULL || now->next == NULL) {
        return NULL;                                           // (2)
    } 
    retNode = now->next;                                       // (3)
    now->next = now->next->next;                               // (4)
    return retNode;
}

struct ListNode* reverseList(struct ListNode* head) {
    struct ListNode *newHead;
    struct ListNode *doRemoveNode = head;                      // (5)
    while(doRemoveNode) {                                      // (6)
        newHead = removeNextAndReturn(doRemoveNode);           // (7)
        if(newHead) {                                          // (8)
            newHead->next = head;                              
            head = newHead;                
        }else {
            break;                                             // (9)
        }
    }
    return head;
}

struct ListNode* merge(struct ListNode* a, struct ListNode* b) {
    struct ListNode* tmp;
    if(a == NULL) {
        return b;
    }
    tmp = a->next;                                            // (10)
    a->next = b;                                              // (11)
    b->next = merge(tmp, b->next);                            // (12)
    return a;
}

void reorderList(struct ListNode* head){
    struct ListNode* half = getHalf(head);
    if(head == half)
        return head;                                         // (13)
    half = reverseList(half);                                // (14)
    merge(head, half);                                       // (15)
}
  • ( 1 ) (1) (1) 删除 now 的 next 结点,并且返回
  • ( 2 ) (2) (2) 本身为空或者下一个结点为空,返回空
  • ( 3 ) (3) (3) 将需要删除的结点缓存起来,供后续返回
  • ( 4 ) (4) (4) 执行删除 now->next 的操作
  • ( 5 ) (5) (5) doRemoveNode->next 是将要被删除的结点,所以 doRemoveNode 需要被缓存起来,不然都不知道怎么进行删除
  • ( 6 ) (6) (6) 没有需要删除的结点了
  • ( 7 ) (7) (7) 删除 doRemoveNode 的下一个节点并返回被删除的结点
  • ( 8 ) (8) (8) 如果有被删除的结点,则插入头部
  • ( 9 ) (9) (9) 如果没有,则跳出迭代
  • ( 10 ) (10) (10) 简单缓存一下
  • ( 11 ) (11) (11) 将 a 指向 b
  • ( 12 ) (12) (12) 将 b 指向 a->next 和 b->next 合并后的结果
  • ( 13 ) (13) (13) 只有一个元素的情况
  • ( 14 ) (14) (14) 执行链表反转
  • ( 15 ) (15) (15) 合并两个链表

三、本题小知识

  合并链表可以采用递归实现,容易理解。


四、加群须知

  相信看我文章的大多数都是「 大学生 」,能上大学的都是「 精英 」,那么我们自然要「 精益求精 」,如果你还是「 大一 」,那么太好了,你拥有大把时间,当然你可以选择「 刷剧 」,然而,「 学好算法 」,三年后的你自然「 不能同日而语 」
  那么这里,我整理了「 几十个基础算法 」 的分类,点击开启:

🌌《算法入门指引》🌌

  如果链接被屏蔽,或者有权限问题,可以私聊作者解决。

  大致题集一览:





  为了让这件事情变得有趣,以及「 照顾初学者 」,目前题目只开放最简单的算法 「 枚举系列 」 (包括:线性枚举、双指针、前缀和、二分枚举、三分枚举),当有 一半成员刷完 「 枚举系列 」 的所有题以后,会开放下个章节,等这套题全部刷完,你还在群里,那么你就会成为「 夜深人静写算法 」专家团 的一员。
  不要小看这个专家团,三年之后,你将会是别人 望尘莫及 的存在。如果要加入,可以联系我,考虑到大家都是学生, 没有「 主要经济来源 」,在你成为神的路上,「 不会索取任何 」
  🔥联系作者,或者扫作者主页二维码加群,加入刷题行列吧🔥


🔥让天下没有难学的算法🔥

C语言免费动漫教程,和我一起打卡!
🌞《光天化日学C语言》🌞

入门级C语言真题汇总
🧡《C语言入门100例》🧡

几张动图学会一种数据结构
🌳《画解数据结构》🌳

组团学习,抱团生长
🌌《算法入门指引》🌌

竞赛选手金典图文教程
💜《夜深人静写算法》💜

以上是关于⭐算法入门⭐《链表》中等02 —— LeetCode 143. 重排链表的主要内容,如果未能解决你的问题,请参考以下文章

⭐算法入门⭐《二叉树》中等01 —— LeetCode 114. 二叉树展开为链表

⭐算法入门⭐《前缀和》中等02 —— LeetCode 974. 和可被 K 整除的子数组

Leetcod-反转链表

⭐算法入门⭐《二分枚举》中等02 —— LeetCode 面试题 10.09. 排序矩阵查找

LeetCode热题100

⭐算法入门⭐《线性枚举》中等02 —— LeetCode 628. 三个数的最大乘积