[LeetCode 61] 旋转链表

题目描述

给定一个链表,旋转链表,将链表每个节点向右移动 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
示例 2:

输入: 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

题目分析

这道题看见旋转链表就想到了三指针,这是旋转链表的万能做法。一个头指针,一个尾指针,一个移动的指针,我也不知道叫啥第三个指针,不过就是用来储存临时的链表并移动吧。也可以说是滑块吧。

源码

/**
* Definition for singly-linked list.
* public class ListNode {
*     int val;
*     ListNode next;
*     ListNode(int x) { val = x; }
* }
*/
class Solution{
public ListNode rotateRight(ListNode head, int k) {
      if(head==null||k==0){
         return head;
       }
     ListNode cursor=head;
     ListNode tail=null;
//尾指针
     int length=1;
     while(cursor.next!=null){
//循环 得到总长度 
    cursor=cursor.next;
    length++;
   }
    int loop=length-(k%length);
//得到循环的次数
    tail=cursor;
//指向尾结点
    cursor.next=head;
//改成循环链表
    cursor=head;
//指向头结点
    for(int i=0;i<loop;i++){
//开始循环
    cursor=cursor.next;
    tail=tail.next;
    }
    tail.next=null;
//改成单链表
    return cursor;
//返回当前头
}
}

改进

三指针其实还比较麻烦,那么能不能用双指针呢,如果用的话是省哪一个指针。必须要遍历一下链表,一直到尾。那么就干脆从尾到头遍历吧,上一个方法是从头到尾旋转。这样多创一个链表,然后尾节点开始旋转k次之后,直接把剩下的复制过去就行。

改进代码

class Solution {
public ListNode rotateRight(ListNode head, int k) {
    if(head == null || (k == 0)) return head;
    ListNode tail = head,p = head;
    int len = 1,pos = 0;
    while(tail.next != null){
        tail = tail.next;
        len ++;
    }
    if((pos = k % len ) == 0) return head;
    //找到新头节点前的节点 将其next设置为null
    for (int i = 0;i < len - pos - 1; i++ ){
        p = p.next;
    }
    ListNode res = p.next;
    p.next = null;
    tail.next = head;
    return res;
}
}

分析

第一个时间复杂度为O(n)
第二个时间复杂度为O(n)

难点

这道题难点就是从断开两个链表直接的连接,必须知道head.next是指的什么链子还是数据的点。这样才能一个一个分析下去。

小结

对于链表还是在草稿纸上画下来慢慢分析的比较好。当看见旋转链表的时候,第一反应就是要用三指针,因为这是一个万能解法,而且很好理解。

[1]https://leetcode-cn.com/problems/rotate-list/submissions/

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值