leetcode - 2181. 合并零之间的节点

70 阅读2分钟

题目描述

给你一个链表的头节点 head ,该链表包含由 0 分隔开的一连串整数。链表的 开端 和 末尾 的节点都满足 Node.val == 0 。

对于每两个相邻的 0 ,请你将它们之间的所有节点合并成一个节点,其值是所有已合并节点的值之和。然后将所有 0 移除,修改后的链表不应该含有任何 0 。

返回修改后链表的头节点 head 。

示例 1:

输入: head = [0,3,1,0,4,5,2,0]
输出: [4,11]
解释:
上图表示输入的链表。修改后的链表包含:

1. 标记为绿色的节点之和:3 + 1 = 4
2. 标记为红色的节点之和:4 + 5 + 2 = 11

示例 2:

输入: head = [0,1,0,3,0,2,2,0]
输出: [1,3,4]
解释:
上图表示输入的链表。修改后的链表包含:

1. 标记为绿色的节点之和:1 = 1
2. 标记为红色的节点之和:3 = 3
3. 标记为黄色的节点之和:2 + 2 = 4

提示:

  • 列表中的节点数目在范围 [3, 2 * 105] 内
  • 0 <= Node.val <= 1000
  •  存在连续两个 Node.val == 0 的节点
  • 链表的 开端 和 末尾 节点都满足 Node.val == 0

题目分析

简单来说,这道题的核心思路就是:

  1. 以值为0的节点作为分隔点,将所有区间内的节点进行值的合并
  2. 将所有值为0的节点去掉;
  3. 按照原有顺序返回新链表的head

思路

双指针来模拟每个区间的头和尾,在指针到达区间尾部之前不断的累加,到达区间尾部时,将区间合并至加入结果链表中。

代码

/**
 * Definition for singly-linked list.
 * class ListNode {
 *     val: number
 *     next: ListNode | null
 *     constructor(val?: number, next?: ListNode | null) {
 *         this.val = (val===undefined ? 0 : val)
 *         this.next = (next===undefined ? null : next)
 *     }
 * }
 */

function mergeNodes(head: ListNode | null): ListNode | null {
    let slow = head, fast = head.next;
    const ans = new ListNode();
    let res = ans;

    while (fast) {
        // 先利用快指针找到当前区间的尾部
        while (fast.val !== 0) {
            fast = fast.next;
        }
        // 将慢指针后移一步到达需要累加的第一个节点的位置
        slow = slow.next;
        // 此时累加时无需考虑后面是否区间尾部,因为尾部为0
        let val = slow.next.val + slow.val;
        // 对于区间内的值进行累加,并且不断后移指针
        while (slow?.next?.val !== 0) {
            slow = slow.next;
            val += slow.next.val;
            slow.val = val;
        }
        // 判断到达区间尾部时,将当前节点加入到结果链表中
        if (slow.next.val === 0 && slow.val !== 0) {
            res.next = slow;
            res = res.next;
        }
        // 短链,后移指针
        slow.next = slow.next.next;
        fast = fast.next;
    }
    
    return ans.next;

};

运行结果

复杂度

  • 时间复杂度: O(n)
  • 空间复杂度: O(1)