问题:
Reverse a linked list from position m to n. Do it in-place and in one-pass.
For example:
Given 1->2->3->4->5->NULL
, m = 2 and n = 4,
return 1->4->3->2->5->NULL
.
Note:
Given m, n satisfy the following condition:
1 ≤ m ≤ n ≤ length of list.
解决:
【题意】把链表中m到n的部分反转(1<=m<=n<=length)。要求:在原地反转,也就是额外的空间O(1),且只能遍历一遍。
①分为两个步骤,第一步是找到m结点所在位置,第二步就是进行反转直到n结点。反转的方法就是每读到一个结点,把它插入到m结点前面位置,然后m结点接到读到结点的下一个。总共只需要一次扫描,所以时间是O(n),只需要几个辅助指针,空间是O(1)。
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode(int x) { val = x; }
* }
*/
class Solution {
public ListNode reverseBetween(ListNode head, int m, int n){
ListNode header = new ListNode(-1);
header.next = head;
ListNode pre = header;
int i = 1;
while(pre.next != null && i < m){
pre = pre.next;
i ++;
}
ListNode mNode = pre.next;
ListNode cur = mNode.next;
while(cur != null && i < n){
mNode.next = cur.next;
cur.next = pre.next;
pre.next = cur;
cur = mNode.next;
i ++;
}
return header.next;
}
}
② 使用for循环。
class Solution { //4ms
public ListNode reverseBetween(ListNode head, int m, int n){
ListNode header = new ListNode(-1);
header.next = head;
ListNode pre = header;
for (int i = 1;i < m;i ++){
pre = pre.next;
}
ListNode mNode = pre.next;
ListNode cur = mNode.next;
for (int i = 0;i < n - m;i ++){
mNode.next = cur.next;
cur.next = pre.next;
pre.next = cur;
cur = mNode.next;
}
return header.next;
}
}