力扣刷题笔记(六)
25. K 个一组翻转链表
难度:困难⭐️⭐️⭐️⭐️⭐️
题目
给你链表的头节点 head ,每 k 个节点一组进行翻转,请你返回修改后的链表。
k 是一个正整数,它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。
你不能只是单纯的改变节点内部的值,而是需要实际进行节点交换。
示例 1:

输入:head = [1,2,3,4,5], k = 2
输出:[2,1,4,3,5]
示例 2:

输入:head = [1,2,3,4,5], k = 3
输出:[3,2,1,4,5]
提示:
- 链表中的节点数目为 n
- 1 <= k <= n <= 5000
- 0 <= Node.val <= 1000
进阶:你可以设计一个只用 O(1) 额外内存空间的算法解决此问题吗?
代码
- 递归
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution {
public ListNode reverseKGroup(ListNode head, int k) {
if (head == null) {
return null;
}
ListNode a, b;
a = b = head;
for (int i = 0; i < k; i++) {
if (b == null) {
return head;
}
b = b.next;
}
ListNode newHead = reverse(a, b);
a.next = reverseKGroup(b, k);
return newHead;
}
private ListNode reverse(ListNode a, ListNode b) {
ListNode pre = null;
ListNode cur = a;
ListNode nxt = cur.next;
while (nxt != b) {
cur.next = pre;
pre = cur;
cur = nxt;
nxt = cur.next;
}
cur.next = pre;
return cur;
}
}
- 模拟
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution {
public ListNode reverseKGroup(ListNode head, int k) {
ListNode dummy = new ListNode();
dummy.next = head;
ListNode pre = dummy;
ListNode end = dummy.next;
while (end != null) {
for (int i = 1; i < k && end != null; i++) {
end = end.next;
}
// 最后不够k个不翻转。恰好够k个的时候,end肯定有值
if (end == null) {
break;
}
// 先断开
ListNode nxt = end.next;
end.next = null;
// 翻转并重新接上
ListNode start = pre.next;
// 翻转并接上左边
pre.next = reverse(start);
// 接上右边,翻转后,start指向的结点跑到右边了
start.next = nxt;
// 还要更新pre
pre = start;
// end向右移
end = nxt;
}
return dummy.next;
}
private ListNode reverse(ListNode a) {
ListNode pre = null;
ListNode cur = a;
ListNode nxt = cur.next;
while (nxt != null) {
cur.next = pre;
pre = cur;
cur = nxt;
nxt = cur.next;
}
cur.next = pre;
return cur;
}
}
笔记
-
主要是控制好指针不要丢了,模拟进行即可。加一个哑巴结点,这样无论之后链表有没有翻转,它的 next 指针都会指向正确的头节点。那么我们只要返回它的下一个节点就好了。
我开源了一份武林秘籍,欢迎⭐️star:
创作不易,喜欢的话加个关注点个赞,❤谢谢谢谢❤
文章介绍了如何使用递归和模拟方法解决LeetCode上的K个一组翻转链表问题,通过控制指针操作实现链表节点的翻转,同时讨论了如何在O(1)额外内存空间下解决该问题。
1666

被折叠的 条评论
为什么被折叠?



