反转从位置 m 到 n 的链表。请使用一趟扫描完成反转。
说明:
1 ≤ m ≤ n ≤ 链表长度。
示例:
输入: 1->2->3->4->5->NULL, m = 2, n = 4
输出: 1->4->3->2->5->NULL
反转链表普通做法头插法(一次遍历)
反转链表普通做法
![在这里插入图片描述](https://img-blog.csdnimg.cn/814954378d5e412f86f67d724160d419.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA56eA56eA55qE5aWH5aaZ5peF6KGM,size_20,color_FFFFFF,t_70,g_se,x_16)
定义指针 rightnode、leftnode、pre、succ、dummynode找到各个指针的位置(for循环)断链、反转、衔接
class Solution {
public ListNode reverseBetween(ListNode head, int left, int right) {
// 因为头节点有可能发生变化,使用虚拟头节点可以避免复杂的分类讨论
ListNode dummyNode = new ListNode(-1);
dummyNode.next = head;
ListNode pre = dummyNode;
// 第 1 步:从虚拟头节点走 left - 1 步,来到 left 节点的前一个节点
// 建议写在 for 循环里,语义清晰
for (int i = 0; i
rightNode = rightNode.next;
}
// 第 3 步:切断出一个子链表(截取链表)
ListNode leftNode = pre.next;
ListNode succ = rightNode.next;
// 注意:切断链接
pre.next = null;
rightNode.next = null;
// 第 4 步:同第 206 题,反转链表的子区间
reverseLinkedList(leftNode);
// 第 5 步:接回到原来的链表中
pre.next = rightNode;
leftNode.next = succ;
return dummyNode.next;
}
private void reverseLinkedList(ListNode head) {
// 也可以使用递归反转一个链表
ListNode pre = null;
ListNode cur = head;
while (cur != null) {
ListNode next = cur.next;
cur.next = pre;
pre = cur;
cur = next;
}
}
}
头插法(一次遍历)
![在这里插入图片描述](https://img-blog.csdnimg.cn/f3644e678cc940a58687ae92b8fe7a13.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA56eA56eA55qE5aWH5aaZ5peF6KGM,size_20,color_FFFFFF,t_70,g_se,x_16)
pre:指向的位置和数值都不变cur:指向的节点不变,位置会变化next:就是头插法要插的那个节点
建dummynode节点 (针对于left是头结点的时候)for循环,找到pre节点 :pre指向了dummynode,为什么还是left-1呢:left指的位置,并不是下标,也就是说left 3 (第三个)位置的节点,下标是left-1 (下标为2)开始反转(头插) 定义next 把要头插的节点摘出来 头插的两句
![在这里插入图片描述](https://img-blog.csdnimg.cn/d78b758e49ed4ff7befc7e62d0bbc955.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA56eA56eA55qE5aWH5aaZ5peF6KGM,size_20,color_FFFFFF,t_70,g_se,x_16)
class Solution {
public ListNode reverseBetween(ListNode head, int left, int right) {
ListNode dummynode=new ListNode(-1);
dummynode.next=head;
ListNode pre=dummynode;
for(int i=0;i
ListNode tmp=cur.next;
cur.next=cur.next.next;
tmp.next=pre.next;
pre.next=tmp;
}
return dummynode.next;
}
}
class Solution {
public ListNode reverseBetween(ListNode head, int left, int right) {
//建虚拟头结点
ListNode dummynode=new ListNode(-1);
dummynode.next=head;
//定义pre,for循环找到pre之后在整个过程中是不变的
ListNode pre=dummynode;
for(int i=0;i
ListNode tmp=cur.next;
//把cur.next摘出来
cur.next=cur.next.next;
//tmp头插法
tmp.next=pre.next;
pre.next=tmp;
}
return dummynode.next;
}
}
![在这里插入图片描述](https://img-blog.csdnimg.cn/dae4a66a72224166acafdf1e4c883ba8.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA56eA56eA55qE5aWH5aaZ5peF6KGM,size_20,color_FFFFFF,t_70,g_se,x_16)
对于链表的问题,根据以往的经验一般都是要建一个dummy node,连上原链表的头结点,这样的话就算头结点变动了,我们还可以通过dummy->next来获得新链表的头结点。这道题的要求是只通过一次遍历完成,就拿题目中的例子来说,变换的是2,3,4这三个点,我们需要找到第一个开始变换结点的前一个结点,只要让pre向后走m-1步即可,为啥要减1呢,因为题目中是从1开始计数的,这里只走了1步,就是结点1,用pre指向它。万一是结点1开始变换的怎么办,这就是我们为啥要用dummy结点了,pre也可以指向dummy结点。然后就要开始交换了,由于一次只能交换两个结点,所以我们按如下的交换顺序:
1 -> 2 -> 3 -> 4 -> 5 -> NULL
1 -> 3 -> 2 -> 4 -> 5 -> NULL
1 -> 4 -> 3 -> 2 -> 5 -> NULL
我们可以看出来,总共需要n-m步即可,第一步是将结点3放到结点1的后面,第二步将结点4放到结点1的后面。这是很有规律的操作,那么我们就说一个就行了,比如刚开始,pre指向结点1,cur指向结点2,然后我们建立一个临时的结点t,指向结点3(注意我们用临时变量保存某个结点就是为了首先断开该结点和前面结点之间的联系,这可以当作一个规律记下来),然后我们断开结点2和结点3,将结点2的next连到结点4上,也就是 cur->next = t->next,再把结点3连到结点1的后面结点(即结点2)的前面,即 t->next = pre->next,最后再将原来的结点1和结点2的连接断开,将结点1连到结点3,即 pre->next = t。这样我们就完成了将结点3取出,加入结点1的后方。第二步将结点4取出,加入结点1的后方,也是同样的操作,
|