🚩传送门:牛客题目
题目
给你一个链表,每 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;}}
