🚩传送门:牛客题目
题目
给你一个链表,每 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]
示例 3:
输入:head = [1,2,3,4,5], k = 1 输出:[1,2,3,4,5]
解题思路:模拟
dummy 头结点
复杂度分析
时间复杂度:,其中
是链表的长度。
- 指针会在
结点上停留,每次停留需要进行一次
的翻转操作
- 综上所述:平均时间复杂度为:
。
空间复杂度: ,我们只需要建立常数个变量。
我的代码
package array.array66;
public class Solution {
public static class ListNode {
int val;
ListNode next;
ListNode() {}
ListNode(int val) { this.val = val; }
ListNode(int val, ListNode next) { this.val = val; this.next = next; }
}
//反转[start,end]切割开的独立的区间段
public static ListNode[] ReviseListNode(ListNode start, ListNode end){
ListNode newHead=start;
ListNode newTail=start;
ListNode curNode=start.next;
ListNode next=null;
start.next=null; //断开首节点
end.next=null; //断开尾节点
while(curNode!=null){ //正常反转
next=curNode.next;
curNode.next=newHead;
newHead=curNode;
curNode=next;
}
return new ListNode[]{newHead,newTail};
}
public static ListNode reverseKGroup(ListNode head, int k) {
//1.合法性判断
if(k==1||head==null) return head;
//2.先new出一个总体的头结点
ListNode dummy=new ListNode(-1);
dummy.next=head;
ListNode pre=dummy;
ListNode next=head;
ListNode start=null;
ListNode end=null;
while(next!=null){
//3.计算[start,end]区间前的初始化
int n=0;
start=next;
end=null; //end=null很关键,是否赋值判断长度够不够 k 的依据
while(next!=null){
n++;
if(n==k){ //长度够 k 个
end=next;
next=next.next;
break;
}
next=next.next;
}
if(end==null){ //未被赋值说明剩下区间不够 k 个
pre.next=start;
}else{ //说明 [start,end] 区间要反转
ListNode[]res= ReviseListNode(start,end);
pre.next=res[0];
pre=res[1]; //return ListNode[]{newHead,newTail};
}
}
//去除头节点返回
return dummy.next;
}
}