相交链表
题目
给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回 null 。
图示两个链表在节点 c1 开始相交:
题目数据 保证 整个链式结构中不存在环。
注意,函数返回结果后,链表必须 保持其原始结构 。
自定义评测:
评测系统 的输入如下(你设计的程序 不适用 此输入):
intersectVal - 相交的起始节点的值。如果不存在相交节点,这一值为 0
listA - 第一个链表
listB - 第二个链表
skipA - 在 listA 中(从头节点开始)跳到交叉节点的节点数
skipB - 在 listB 中(从头节点开始)跳到交叉节点的节点数
评测系统将根据这些输入创建链式数据结构,并将两个头节点 headA 和 headB 传递给你的程序。如果程序能够正确返回相交节点,那么你的解决方案将被 视作正确答案 。
示例 1:
1 2 3 4 5 输入:intersectVal = 8 , listA = [4,1,8,4 ,5 ], listB = [5,6,1,8 ,4 ,5 ], skipA = 2 , skipB = 3 输出:Intersected at '8 ' 解释:相交节点的值为 8 (注意,如果两个链表相交则不能为 0 )。 从各自的表头开始算起,链表 A 为 [4,1,8,4 ,5 ],链表 B 为 [5,6,1,8 ,4 ,5 ]。 在 A 中,相交节点前有 2 个节点;在 B 中,相交节点前有 3 个节点。
示例 2:
1 2 3 4 5 输入:intersectVal = 2 , listA = [1 ,9 ,1 ,2 ,4 ], listB = [3 ,2 ,4 ], skipA = 3 , skipB = 1 输出:Intersected at '2 ' 解释:相交节点的值为 2 (注意,如果两个链表相交则不能为 0 )。 从各自的表头开始算起,链表 A 为 [1 ,9 ,1 ,2 ,4 ],链表 B 为 [3 ,2 ,4 ]。 在 A 中,相交节点前有 3 个节点;在 B 中,相交节点前有 1 个节点。
示例 3:
1 2 3 4 5 输入:intersectVal = 0 , listA = [2 ,6 ,4 ], listB = [1 ,5 ], skipA = 3 , skipB = 2 输出:null 解释:从各自的表头开始算起,链表 A 为 [2 ,6 ,4 ],链表 B 为 [1 ,5 ]。 由于这两个链表不相交,所以 intersectVal 必须为 0 ,而 skipA 和 skipB 可以是任意值。 这两个链表不相交,因此返回 null 。
提示:
listA 中节点数目为 m
listB 中节点数目为 n
1 <= m, n <= 3 * 104
1 <= Node.val <= 105
0 <= skipA <= m
0 <= skipB <= n
如果 listA 和 listB 没有交点,intersectVal 为 0
如果 listA 和 listB 有交点,intersectVal == listA[skipA] == listB[skipB]
题解
双指针
考虑构建两个节点指针 A , B 分别指向两链表头节点 headA , headB ,做如下操作:
指针 A 先遍历完链表 headA ,再开始遍历链表 headB ,当走到 node 时,共走步数为:
指针 B 先遍历完链表 headB ,再开始遍历链表 headA ,当走到 node 时,共走步数为:
如下式所示,此时指针 A , B 重合,并有两种情况:
a + (b - c) = b + (a - c)
若两链表 有 公共尾部 (即 c > 0 ) :指针 A , B 同时指向「第一个公共节点」node 。
若两链表 无 公共尾部 (即 c = 0 ) :指针 A , B 同时指向 null 。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 class Solution : def getIntersectionNode (self, headA: ListNode, headB: ListNode ) -> ListNode: pa, pb = headA,headB ra,rb = False ,False while True : if pa==pb:return pa if pa.next :pa = pa.next else : if not ra:pa, ra = headB, True else :return None if pb.next :pb = pb.next else : if not rb: pb, rb = headA, True else :return None
简约版:
1 2 3 4 5 6 7 8 class Solution : def getIntersectionNode (self, headA: ListNode, headB: ListNode ) -> ListNode: A, B = headA, headB while A != B: A = A.next if A else headB B = B.next if B else headA return A