LeetCode 题解:25. K 个一组翻转链表,迭代,JavaScript,详细注释
发布于: 2020 年 08 月 23 日

原题链接:https://leetcode-cn.com/problems/reverse-nodes-in-k-group/
解题思路:
可以先参考官方题解,再看本题解。
总体思路如下:
将链表拆分成N个子链表,每个子链表有K个节点。
单独翻转每个子链表。
将翻转后的子链表与前后节点连接起来。
需要注意判断链表为null,以及链表节点总数不是 k 的整数倍的情况。
如果说一次写出完整代码比较困难,我们可以分步骤解题:
写出循环链表方法
/** * @param {ListNode} head * @param {number} k * @return {ListNode} */var reverseKGroup = function (head, k) {  // 反转子链表的方法,在 206. 反转链表 的基础上做改造,可以参考如下题解:  // https://leetcode-cn.com/problems/reverse-linked-list/solution/206-fan-zhuan-lian-biao-javascriptwhilexun-huan-di/  function reverseNode(head, tail) {    // 待第2步补充  }  // 创建dummy节点,将其作为第一次遍历链表时的prevNode  // dummyNode不会参与链表遍历,也就是说它的next指向始终是链表的头结点  let dummyNode = new ListNode(-1);  dummyNode.next = head;  // 将dummyNode赋值给prevNode进入链表的遍历。  // prevNode在遍历时,会不断移动。  // 在第一次遍历时,prevNode.next会指向翻转后的链表头节点。  // 此时dummyNode.next的指向也会同时被改变。  let prevNode = dummyNode;  while (head) {    // 创建一个尾节点,它会被移动k位,最终指向当前子链表的伪节点。。    let tail = prevNode;    // 移动tail到当前子链表的尾部,同时判断剩余节点的长度是否>=k    for (let i = 0; i < k; i++) {      // 待第3步补充    }    // nextNode即为下一段子链表的头节点    let nextNode = tail.next;    // 将子链表反转,返回其新的头尾节点,用于连接符链表    [head, tail] = reverseNode(head, tail);    // 将父链表与子链表链接    prevNode.next = head;    tail.next = nextNode;    // 向前移动prevNode和head,继续下一个子链表的循环    prevNode = tail;    head = tail.next;  }  // dummyNode的指向是链表的头节点,输出dummyNode.next就是将翻转后的链表输出。  // 如果head为null,则不户进入while循环,而是直接被返回  return dummyNode.next;};
增加反转子链表方法
/** * @param {ListNode} head * @param {number} k * @return {ListNode} */var reverseKGroup = function (head, k) {  // 反转子链表的方法,在 206. 反转链表 的基础上做改造,可以参考如下题解:  // https://leetcode-cn.com/problems/reverse-linked-list/solution/206-fan-zhuan-lian-biao-javascriptwhilexun-huan-di/  function reverseNode(head, tail) {    // 如果子链表已经反转,那个反转之后的签一个节点,即为当前子链表的下一个节点。    // 因此直接把签一个节点设置为tail.next    let prevNode = tail.next;    // 缓存head,使用node遍历子链表并进行反转。    // 因为head指针不能改变,需要用于返回反转后的尾结点。    let node = head;    // 进行链表反转操作,prevNode不断移动,移动到tail时,表示该子链表反转完成。    while (prevNode !== tail) {      const tempNode = node.next;      node.next = prevNode;      prevNode = node;      node = tempNode;    }    // 完成反转后,原来的头尾节点已经对调,则反过来返回。    // 要注意这里,头尾节点其实都没有改变,只是他们的位置因为子链表被反转而调换了。    return [tail, head];  }  // 创建dummy节点,将其作为第一次遍历链表时的prevNode  // dummyNode不会参与链表遍历,也就是说它的next指向始终是链表的头结点  let dummyNode = new ListNode(-1);  dummyNode.next = head;  // 将dummyNode赋值给prevNode进入链表的遍历。  // prevNode在遍历时,会不断移动。  // 在第一次遍历时,prevNode.next会指向翻转后的链表头节点。  // 此时dummyNode.next的指向也会同时被改变。  let prevNode = dummyNode;  while (head) {    // 创建一个尾节点,它会被移动k位,最终指向当前子链表的伪节点。。    let tail = prevNode;    // 移动tail到当前子链表的尾部,同时判断剩余节点的长度是否>=k    for (let i = 0; i < k; i++) {      // 待第3步补充    }    // nextNode即为下一段子链表的头节点    let nextNode = tail.next;    // 将子链表反转,返回其新的头尾节点,用于连接符链表    [head, tail] = reverseNode(head, tail);    // 将父链表与子链表链接    prevNode.next = head;    tail.next = nextNode;    // 向前移动prevNode和head,继续下一个子链表的循环    prevNode = tail;    head = tail.next;  }  // dummyNode的指向是链表的头节点,输出dummyNode.next就是将翻转后的链表输出。  // 如果head为null,则不户进入while循环,而是直接被返回  return dummyNode.next;};
判断剩余节点的长度是否>=k
/** * @param {ListNode} head * @param {number} k * @return {ListNode} */var reverseKGroup = function (head, k) {  // 反转子链表的方法,在 206. 反转链表 的基础上做改造,可以参考如下题解:  // https://leetcode-cn.com/problems/reverse-linked-list/solution/206-fan-zhuan-lian-biao-javascriptwhilexun-huan-di/  function reverseNode(head, tail) {    // 如果子链表已经反转,那个反转之后的签一个节点,即为当前子链表的下一个节点。    // 因此直接把签一个节点设置为tail.next    let prevNode = tail.next;    // 缓存head,使用node遍历子链表并进行反转。    // 因为head指针不能改变,需要用于返回反转后的尾结点。    let node = head;    // 进行链表反转操作,prevNode不断移动,移动到tail时,表示该子链表反转完成。    while (prevNode !== tail) {      const tempNode = node.next;      node.next = prevNode;      prevNode = node;      node = tempNode;    }    // 完成反转后,原来的头尾节点已经对调,则反过来返回。    // 要注意这里,头尾节点其实都没有改变,只是他们的位置因为子链表被反转而调换了。    return [tail, head];  }  // 创建dummy节点,将其作为第一次遍历链表时的prevNode  // dummyNode不会参与链表遍历,也就是说它的next指向始终是链表的头结点  let dummyNode = new ListNode(-1);  dummyNode.next = head;  // 将dummyNode赋值给prevNode进入链表的遍历。  // prevNode在遍历时,会不断移动。  // 在第一次遍历时,prevNode.next会指向翻转后的链表头节点。  // 此时dummyNode.next的指向也会同时被改变。  let prevNode = dummyNode;  while (head) {    // 创建一个尾节点,它会被移动k位,最终指向当前子链表的伪节点。。    let tail = prevNode;    // 移动tail到当前子链表的尾部,同时判断剩余节点的长度是否>=k    for (let i = 0; i < k; i++) {      // 向后移动tail指针      // 注意这里要先移动,再进行是否为空的判断,否则会报错      tail = tail.next;      // 如果tail为null,说明链表剩余部分不足k个,无需继续反转,直接返回结果      if (!tail) {        return dummyNode.next;      }    }    // nextNode即为下一段子链表的头节点    let nextNode = tail.next;    // 将子链表反转,返回其新的头尾节点,用于连接符链表    [head, tail] = reverseNode(head, tail);    // 将父链表与子链表链接    prevNode.next = head;    tail.next = nextNode;    // 向前移动prevNode和head,继续下一个子链表的循环    prevNode = tail;    head = tail.next;  }  // dummyNode的指向是链表的头节点,输出dummyNode.next就是将翻转后的链表输出。  // 如果head为null,则不户进入while循环,而是直接被返回  return dummyNode.next;};
 划线
   评论
  复制
发布于: 2020 年 08 月 23 日 阅读数: 27
版权声明: 本文为 InfoQ 作者【Lee Chen】的原创文章。
原文链接:【http://xie.infoq.cn/article/ef6ad38fc44bf952e482658bc】。文章转载请联系作者。
Lee Chen
  关注 
还未添加个人签名 2018.08.29 加入
还未添加个人简介
 
 
  
  
 
 
 
 
 
 
 
 
 
 
    
评论