⭐算法入门⭐《链表》中等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 L0→L1→...→Ln−1→Ln请将其重新排列后变为: 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 ... L0→Ln→L1→Ln−1→L2→Ln−2→...不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。
样例输入:head = [1,2,3,4,5,6,7]
样例输出:[1,7,2,6,3,5,4]
2、基础框架
- C语言 版本给出的基础框架代码如下:
void reorderList(struct ListNode* head) {
}
3、原题链接
二、解题报告
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 整除的子数组