【LeetCode 剑指Offer 52】链表相交 ,两个链表的第一个公共结点,给定两个(单向)链表,判定它们是否相交并返回交点。请注意相交的定义基于节点的引用,而不是基于节点的值

学习目标:

目标:熟练运用Java所学知识


学习内容:

本文内容:使用Java实现:链表相交


题目描述:

给定两个(单向)链表,判定它们是否相交并返回交点。请注意相交的定义基于节点的引用,而不是基于节点的值。换句话说,如果一个链表的第k个节点与另一个链表的第j个节点是同一节点(引用完全相同),则这两个链表相交。

输入两个链表,找出它们的第一个公共节点。

如下面的两个链表:

在这里插入图片描述

在节点 c1 开始相交。

示例 1:

在这里插入图片描述

输入:intersectVal = 8, listA = [4,1,8,4,5], listB = [5,0,1,8,4,5], skipA = 2, skipB = 3
输出:Reference of the node with value = 8
输入解释:相交节点的值为 8 (注意,如果两个列表相交则不能为 0)。从各自的表头开始算起,链表 A 为 [4,1,8,4,5],链表 B 为 [5,0,1,8,4,5]。在 A 中,相交节点前有 2 个节点;在 B 中,相交节点前有 3
个节点。

示例 2:
在这里插入图片描述

输入:intersectVal = 2, listA = [0,9,1,2,4], listB = [3,2,4], skipA = 3, skipB = 1
输出:Reference of the node with value = 2
输入解释:相交节点的值为 2 (注意,如果两个列表相交则不能为 0)。从各自的表头开始算起,链表 A 为 [0,9,1,2,4],链表 B 为 [3,2,4]。在 A 中,相交节点前有 3 个节点;在 B 中,相交节点前有 1 个节点。

示例 3:
在这里插入图片描述

输入: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。

解题思路

如果两条链表相交,则交点之后的结点一定也是公共结点,如下图所示:
在这里插入图片描述
解释:由于单向链表只有一个指向下一个结点的指针next,所以只要有交点,则两条链表之后的结点都是公共结点。

重要思想:想要判断两条链表是否相交,只需要判断出有一个公共结点即可,很容易想到使用双重循环,逐个的排除,但是这样太浪费时间,由于两条链表相交交点之后的长度是一样的,所以有比较简单的做法:

  • 首先求出两个链表长度
     int countA=0,countB=0;
       ListNode curA=headA,curB=headB;
       //循环遍历,求出链表长度
       while(curA!=null){
           curA=curA.next;
           countA++;
       }
       while(curB!=null){
           curB=curB.next;
           countB++;
       }
       //将curA和curB的位置更新至表头
       curA=headA;
       curB=headB;
  • 判断哪个链表较长,求出长度差
  • 将较长的链表从头向后遍历长度差个节点
//判断长度,将较长的链表向后遍历长度差个节点
    if(countA>countB){
           for(int i=0;i<countA-countB;i++){
               curA=curA.next;
           }

       }else{
           for(int i=0;i<countB-countA;i++){
               curB=curB.next;
           }
       }
  • 然后同时遍历两个链表,判断引用是否相等
//循环遍历,判断节点引用是否相同
        while(curA!=null&&curB!=null){
            if(curA==curB){
                return curA;//如果相同则返回该结点
            }
            curA=curA.next;
            curB=curB.next;
        }
        return null;//跳出while循环说明没有公共结点

实现代码

    public static ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        int countA=0,countB=0;
        ListNode curA=headA,curB=headB;
        //循环遍历,求出链表长度
        while(curA!=null){
            curA=curA.next;
            countA++;
        }
        while(curB!=null){
            curB=curB.next;
            countB++;
        }
        curA=headA;
        curB=headB;
//判断长度,将较长的链表向后遍历长度差个节点
        if(countA>countB){
            for(int i=0;i<countA-countB;i++){
                curA=curA.next;
            }

        }else{
            for(int i=0;i<countB-countA;i++){
                curB=curB.next;
            }
        }
//循环遍历,判断节点引用是否相同
        while(curA!=null&&curB!=null){
            if(curA==curB){
                return curA;//如果相同则返回该结点
            }
            curA=curA.next;
            curB=curB.next;
        }
        return null;//跳出while循环说明没有公共结点
    }
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值