跳转至

21. 合并两个有序链表

题目

将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。

示例 1:

输入:l1 = [1,2,4], l2 = [1,3,4]

输出:[1,1,2,3,4,4]

示例 2:

输入:l1 = [], l2 = []

输出:[]

思路

比较 \(list_1\)\(list_2\) 的节点值,如果 \(list_1\) 的节点值小,则把 \(list_1\) 加到新链表的末尾,然后把 \(list_1\) 替换成它的下一个节点。如果 \(list_2\) 的节点值小则同理。如果两个节点值一样,那么把谁加到新链表的末尾都是一样的,不妨规定把 \(list_2\) 加到新链表末尾。

题解

Go
func mergeTwoLists(l1 *ListNode, l2 *ListNode) *ListNode {
    dummy := &ListNode{}         // 虚拟头结点
    prev := dummy                // 用prev去扫,先指向dummy
    for l1 != nil && l2 != nil { // l1 l2 都存在
        if l1.Val <= l2.Val {    // l1值较小
            prev.Next = l1       // prev.Next指向l1
            l1 = l1.Next         // 考察l1的下一个结点
        } else {
            prev.Next = l2
            l2 = l2.Next
        }
        prev = prev.Next         // prev.Next确定了,prev指针推进
    }

    if l1 != nil {
        prev.Next = l1
    }

    if l2 != nil {
        prev.Next = l2
    }

    return dummy.Next
}
Python
class Solution:
    def mergeTwoLists(self, list1: Optional[ListNode], list2: Optional[ListNode]) -> Optional[ListNode]:
        cur = dummy = ListNode()  # 用哨兵节点简化代码逻辑
        while list1 and list2:
            if list1.val < list2.val:
                cur.next = list1  # 把 list1 加到新链表中
                list1 = list1.next
            else:  # 注:相等的情况加哪个节点都是可以的
                cur.next = list2  # 把 list2 加到新链表中
                list2 = list2.next
            cur = cur.next
        cur.next = list1 if list1 else list2  # 拼接剩余链表
        return dummy.next

复杂度

  • 时间复杂度:\(O(n + m)\),其中 \(n\)\(list_1\) 的长度,\(m\)\(list_2\) 的长度。
  • 空间复杂度:\(O(1)\)

参考