旋转链表
Tips
题目
给定一个链表, 旋转链表, 将链表每个节点向右移动 k
个位置, 其中 k
是非负数.
提示:
- 链表中节点的数目在范围
[0, 500]
内 -100 <= Node.val <= 100
0 <= k <= 2 * 10⁹
示例
输入: 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
输入: 0 -> 1 -> 2 -> NULL, k = 4
输出: 2 -> 0 -> 1 -> NULL
解释:
向右旋转 1 步: 2 -> 0 -> 1 -> NULL
向右旋转 2 步: 1 -> 2 -> 0 -> NULL
向右旋转 3 步: 0 -> 1 -> 2 -> NULL
向右旋转 4 步: 2 -> 0 -> 1 -> NULL
题解
- 首先获取整个链表
head
的长度n
- 通过
k = k % n
获取最小旋转的k
, 因为题目有的k
超过了总长度 - 通过快慢指针的方法找到倒数第
k
个节点. 快慢指针的思路是让快指针先走k
步, 再快慢指针往前一起走, 快指针走到头了, 慢指针就指向了链表中倒数第k
个节点, 具体可以看 22. 链表中倒数第 k 个节点 这道题, - 通过
slow.next
获取newHead
- 因为 fast 指向了链表的最后, 所以让
fast.next = head
- 让
slow.next
设为null
, 目的是断开链表.
/**
* Definition for singly-linked list.
* function ListNode(val) {
* this.val = val;
* this.next = null;
* }
*/
/**
* @param {ListNode} head
* @param {number} k
* @return {ListNode}
*/
var rotateRight = function (head, k) {
if (!head || !head.next) return head
let curr = head
let n = 0
while (curr) {
curr = curr.next
n++
}
k = k % n
if (k === 0) return head
let fast = head,
slow = head
while (k) {
fast = fast.next
k--
}
while (fast.next) {
fast = fast.next
slow = slow.next
}
const newHead = slow.next
slow.next = null
fast.next = head
return newHead
}