【面试题25 合并两个排序的链表】
难度: 简单
输入两个递增排序的链表,合并这两个链表并使新链表中的节点仍然是递增排序的。
Leetcode题目对应位置: 面试题25:合并两个排序的链表
思路 1:双指针
题目没有要求不能改变原单链表,所以可以直接在上面进行修改。基本思路是先初始化一个新的头指针,然后比较链表 l1
和 l2
的头指针所指向位置的值的大小关系,将较小一个的链接到头指针后,不断后移 l1
和 l2
,直到某一个链表为空,这时将剩下的链表直接链接到新链表后面即可。
初始化: head
指针指向一个伪头节点,用于创建新链表,用伪节点是为了将第一个结点与后面的结点统一处理
循环过程: 比较 l1
和 l2
当前节点的大小,将较小的加入新链表
循环结束条件: 当遍历到 l1
或 l2
链表尾时,循环结束,将剩余的节点依次添加到新链表中
时间复杂度: O(m + n),m 和 n 分别为两个链表的长度
空间复杂度: O(1),维护两个新指针,分别作为头节点指针和遍历指针
# Definition for singly-linked list.
# class ListNode:
# def __init__(self, x):
# self.val = x
# self.next = None
class Solution:
def mergeTwoLists(self, l1: ListNode, l2: ListNode) -> ListNode:
cur = head = ListNode(0)
while l1 and l2:
if l1.val < l2.val:
cur.next, l1 = l1, l1.next
else:
cur.next, l2 = l2, l2.next
cur = cur.next
cur.next = l1 if l1 else l2
return head.next
代码来源:solution/mian-shi-ti-25-he-bing-liang-ge-pai-xu-de-lian-b-2
如果题目要求不能改变原本的两个链表,就需要另开空间和指针来构造新链表。
思路 2:递归
通过递归自顶向下地进行索引节点:
递归函数功能: 返回 l1
指向的结点和 l2
指向的结点中值较小的结点,并将从下级函数获得的返回值,链接到当前结点尾部
递归边界: 当 l1
为空或 l2
为空,函数结束,返回 l1
或 l2
中剩下的部分。
时间复杂度: O(m + n),m 和 n 分别为两个链表的长度
空间复杂度: O(m + n),最坏情况下递归深度为两链表长度之和
class Solution:
def mergeTwoLists(self, l1: ListNode, l2: ListNode) -> ListNode:
if not l1 and not l2: return
if not l2: return l1
if not l1: return l2
if l1.val <= l2.val:
l1.next = self.mergeTwoLists(l1.next, l2)
return l1
else:
l2.next = self.mergeTwoLists(l1, l2.next)
return l2
代码来源:solution/he-bing-liang-ge-pai-xu-de-lian-biao-die-dai-di-gu