一、剑指 Offer 24. 反转链表 /leetcode 206
题目链接:剑指 Offer 24. 反转链表 - 力扣(LeetCode) (leetcode-cn.com)
题目描述:定义一个函数,输入一个链表的头节点,反转该链表并输出反转后链表的头节点。
方法一:迭代
在遍历链表时,将当前节点的next指针改为指向前一个节点。由于节点没有引用其前一个节点,因此必须事先存储其前一个节点。在更改引用之前,还需存储后一个节点。最后返回新的头引用。
class Solution{
public ListNode reverseList(ListNode head){
ListNode prev =null;
ListNode curr = head;
while(curr!=null){
// 需要存储curr的下一个节点
ListNode next = curr.next;
curr.next= pre;
prev = curr;
curr = next;
}
return prev;
}
}
复杂度分析
- 时间复杂度:O(n)O(n),其中 nn 是链表的长度。需要遍历链表一次。
- 空间复杂度:O(1)O(1)。
方法二:递归
class Solution {
public ListNode reverseList(ListNode head) {
//当参数head为最后一个节点时,无需再继续调用翻转函数,直接返回
if (head == null || head.next == null) {
return head;
}
// 调用返回后,说明从head.next开始到尾结点已经翻转
ListNode newHead = reverseList(head.next);
// 只需考虑翻转后,head节点与已翻转链表如何调整关系
head.next.next = head;
head.next = null;
// 返回翻转后的头节点
return newHead;
}
}
递归:
- 每调用一次递归函数,返回后说明小问题已经解决
- 剩下的代码只需考虑如何在解决小问题的基础上怎么解决稍大的问题
leetcode 92 反转链表II
题目链接:92. 反转链表 II - 力扣(LeetCode) (leetcode-cn.com)
题目描述:给你单链表的头指针 head 和两个整数 left 和 right ,其中 left <= right 。请你反转从位置 left 到位置 right 的链表节点,返回 反转后的链表 。
方法一:穿针引线
思路:
使用上面206的解法,反转left到right部分以后,再拼接起来。还需要记录left的前一个结点和right的后一个节点。
算法步骤:
- 第 1 步:先将待反转的区域反转;
- 第 2 步:把 pre 的 next 指针指向反转以后的链表头节点,把反转以后的链表的尾节点的 next 指针指向 succ
class Solution {
public ListNode reverseBetween(ListNode head, int left, int right) {
// 因为头节点有可能发生变化,使用虚拟头节点可以避免复杂的分类讨论
ListNode dummyNode = new ListNode(-1);
dummyNode.next = head;
ListNode pre = dummyNode;
// 第 1 步:从虚拟头节点走 left - 1 步,来到 left 节点的前一个节点
// 建议写在 for 循环里,语义清晰
for (int i = 0; i < left - 1; i++) {
pre = pre.next;
}
// 第 2 步:从 pre 再走 right - left + 1 步,来到 right 节点
ListNode rightNode = pre;
for (int i = 0; i < right - left + 1; i++) {
rightNode = rightNode.next;
}
// 第 3 步:切断出一个子链表(截取链表)
ListNode leftNode = pre.next;
ListNode curr = rightNode.next;
// 注意:切断链接
pre.next = null;
rightNode.next = null;
// 第 4 步:同第 206 题,反转链表的子区间
reverseLinkedList(leftNode);
// 第 5 步:接回到原来的链表中
pre.next = rightNode;
leftNode.next = curr;
return dummyNode.next;
}
private void reverseLinkedList(ListNode head) {
// 也可以使用递归反转一个链表
ListNode pre = null;
ListNode cur = head;
while (cur != null) {
ListNode next = cur.next;
cur.next = pre;
pre = cur;
cur = next;
}
}
}
方法二:头插法
思路:
在需要反转的区间内,每遍历到一个节点,让这个新节点来到反转部分的起始位置。
class Solution {
public ListNode reverseBetween(ListNode head, int m, int n) {
// 定义一个dummyHead, 方便处理
ListNode dummyHead = new ListNode(0);
dummyHead.next = head;
// 初始化指针
ListNode g = dummyHead;
ListNode p = dummyHead.next;
// 将指针移到相应的位置
for(int step = 0; step < m - 1; step++) {
g = g.next; p = p.next;
}
// 头插法插入节点
for (int i = 0; i < n - m; i++) {
ListNode removed = p.next;
p.next = p.next.next;
removed.next = g.next;
g.next = removed;
}
return dummyHead.next;
}
}