Leetcode刷题之链表小结-1 | 92反转链表 | 206反转链表
小结
1. 如何反转某一个节点的指向?
206反转链表(简单)的递归解法——该方法的理念是: 若节点k+1到节点m已经被反转,而我们当前处于k位置,那么我们希望k+1指向k, 体现在以下代码的head->next->next = head;
这一句,可以记做一种常用的反转单个节点的方法.
class Solution {
public:
ListNode* reverseList(ListNode* head) {
if (!head || !head->next) {
return head;
}
ListNode* newHead = reverseList(head->next);
head->next->next = head;
head->next = nullptr;
return newHead;
}
};
2. 头插法|穿针引线法
在92反转链表(中等难度)题目中, 出现了“穿针引线”、“头插法”等词汇.
题目的要求是反转链表中从left开始到right结束的元素, 在图解中,用橙色部分表示.头插法的理念是,在待反转区域,每遍历到一个节点cur, 都将它的下一个节点next, 放置到cur的前面.
头插法分为两步, 一是“穿针引线”, 二是拉直.
“穿针引线”的过程就是确定这一次操作要确定的各节点位置, 想象第一次动手时, 当前我们拿着一根针在7的位置上, 那么“头”就是5, 接下来穿到2, 再穿到4, 然后拉直这条线, 新的节点就如图2所示了.
拉直以后:
官方的头插法代码:
class Solution {
public:
ListNode *reverseBetween(ListNode *head, int left, int right) {
// 设置 dummyNode 是这一类问题的一般做法
ListNode *dummyNode = new ListNode(-1);
dummyNode->next = head;
ListNode *pre = dummyNode;
for (int i = 0; i < left - 1; i++) {
pre = pre->next;
}
ListNode *cur = pre->next;
ListNode *next;
for (int i = 0; i < right - left; i++) {
next = cur->next;
cur->next = next->next;
next->next = pre->next;
pre->next = next;
}
return dummyNode->next;
}
};