题目地址:https://leetcode-cn.com/problems/rotate-list/
题目:
给定一个链表,旋转链表,将链表每个节点向右移动 k 个位置,其中 k 是非负数。
示例 1:
输入: 1->2->3->4->5->NULL, k = 2
输出: 4->5->1->2->3->NULL
解释:
向右旋转 1 步: 5->1->2->3->4->NULL
向右旋转 2 步: 4->5->1->2->3->NULL
试题分析:
这道题如果前期没有准备过直接写很容易漏掉一些异常处理和移位超出链表长度的处理,总体算法思路是先找到需要旋转的位置,这里有一个点需要注意下,就是移动的位数k有可能会大于链表总长度,所以需要先遍历出链表总长度,然后总长度减去k后按照总长度取模 length-k%length
遍历链表找到要旋转的节点位置,然后定义一个旋转后的链表头指向这个节点的next,然后让旧链表的尾节点和旧链表的头节点对接,这样就完成了旋转。
代码:
public ListNode rotateRight(ListNode head, int k) {
if(head==null || k==0 ||head.next==null){
return head;
}
ListNode p = head;
//遍历出链表长度
int length = 0;
ListNode tail = head;
for(;tail.next!=null;tail=tail.next){
length++;
}
length++;
//找到需要反转的节点的前置节点
for(int i=1;i<length-k%length;i++){
p = p.next;
}
//如果反转节点的前置节点是最后一个节点则不需要反转直接返回原头节点
if(p.next==null){
return head;
}else {
//新翻转后的头节点指向刚才遍历出的反转节点的前置节点的next节点
ListNode newHead = p.next;
//尾节点next指向原head节点
tail.next = head;
p.next = null;
return newHead;
}
}
源码路径:com.monkey01.linkedlist.RotateList_61
配套测试代码路径:test目录com.monkey01.linkedlist.RotateList_61Test