leetcode【链表—简单】07.链表相交

简介: leetcode【链表—简单】07.链表相交

题目


题目来源leetcode


leetcode地址:面试题 02.07. 链表相交,难度:简单。


题目描述(摘自leetcode):


给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表没有交点,返回 null 。
图示两个链表在节点 c1 开始相交:
题目数据 保证 整个链式结构中不存在环。
注意,函数返回结果后,链表必须 保持其原始结构 。
示例 1:
输入:intersectVal = 8, listA = [4,1,8,4,5], listB = [5,0,1,8,4,5], skipA = 2, skipB = 3
输出:Intersected at '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
输出:Intersected at '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 。
提示:
  listA 中节点数目为 m
  listB 中节点数目为 n
  0 <= m, n <= 3 * 104
  1 <= Node.val <= 105
  0 <= skipA <= m
  0 <= skipB <= n
  如果 listA 和 listB 没有交点,intersectVal 为 0
  如果 listA 和 listB 有交点,intersectVal == listA[skipA + 1] == listB[skipB + 1]


本地调试代码:


class ListNode {
    int val;
    ListNode next;
    ListNode() {}
    ListNode(int val) { this.val = val; }
    ListNode(int val, ListNode next) { this.val = val; this.next = next; }
}
class Solution {
    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        ...
    }
    public static void main(String[] args) {
        //公共节点
        ListNode node2 = new ListNode(4, null);
        ListNode node1 = new ListNode(3, node2);
        //链表A
        ListNode aNode2 = new ListNode(2, node1);
        ListNode aNode1 = new ListNode(1, aNode2);
        //链表B
        ListNode bNode1 = new ListNode(1, node1);
        //测试
        ListNode intersectionNode = new Solution().getIntersectionNode(aNode1, bNode1);
        //遍历
        printList(intersectionNode);
    }
    private static void printList(ListNode listNode) {
        if(listNode == null){
            return;
        }
        System.out.print("[");
        while(listNode != null){
            System.out.print(listNode.val);
            if(listNode.next != null){
                System.out.print(",");
            }
            listNode = listNode.next;
        }
        System.out.print("]");
    }
}



题解


NO1:差值移动比较法


思路:题目说要找到对应两个链表中公共的链表头节点,首先我们求出两个链表的长度,让长的链表先移动指定的差值,此时两个链表长度一致,接着两个链表同时向后移动节点进行比对即可找出头节点。


代码:时间复杂度O(n+m)


public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
    int intersectVal = 0;
    //求A、B链表各自的长度
    int aSize = getListSize(headA);
    int bSize = getListSize(headB);
    if(aSize == 0 || bSize == 0){
        return null;
    }
    //长度差值
    int moveStep = Math.abs(aSize-bSize);
    ListNode maxList = null;
    ListNode minList = null;
    if(aSize>bSize){
        maxList = headA;
        minList = headB;
    }else if(aSize<bSize){
        maxList = headB;
        minList = headA;
    }else{
        maxList = headA;
        minList = headB;
    }
    //让长的进行移动差值步
    for (int i = 1; i <= moveStep; i++) {
        maxList = maxList.next;
    }
    //此时大家从同一最后步长开始移动
    while(maxList != null ){//这里任意取某一个为不为空情况
        if(maxList == minList){ //比较的是对象是否相当,是否为同一个节点
            return maxList;
        }
        maxList = maxList.next;
        minList = minList.next;
    }
    //链表不相交情况
    return null;
}
public int getListSize(ListNode head){
    int num = 0;
    while(head != null){
        num++;
        head = head.next;
    }
    return num;
}



相关文章
|
1月前
|
算法
LeetCode第24题两两交换链表中的节点
这篇文章介绍了LeetCode第24题"两两交换链表中的节点"的解题方法,通过使用虚拟节点和前驱节点技巧,实现了链表中相邻节点的交换。
LeetCode第24题两两交换链表中的节点
|
1月前
|
存储 算法
LeetCode第86题分隔链表
文章介绍了LeetCode第86题"分隔链表"的解法,通过创建两个新链表分别存储小于和大于等于给定值x的节点,然后合并这两个链表来解决问题,提供了一种简单易懂且操作原链表的解决方案。
LeetCode第86题分隔链表
|
1月前
|
存储 算法
LeetCode第83题删除排序链表中的重复元素
文章介绍了LeetCode第83题"删除排序链表中的重复元素"的解法,使用双指针技术在原链表上原地删除重复元素,提供了一种时间和空间效率都较高的解决方案。
LeetCode第83题删除排序链表中的重复元素
|
1月前
|
算法
LeetCode第23题合并 K 个升序链表
这篇文章介绍了LeetCode第23题"合并K个升序链表"的解题方法,使用分而治之的思想,通过递归合并链表的方式解决了这个难题。
LeetCode第23题合并 K 个升序链表
|
1月前
|
C++ 索引
leetcode 707.设计链表
本文提供了解决LeetCode 707题"设计链表"的C++实现,包括单链表的节点定义和类方法实现,如添加节点、获取节点值、删除节点等。
|
1月前
|
算法
LeetCode第92题反转链表 II
文章分享了LeetCode第92题"反转链表 II"的解法,通过使用四个指针来记录和更新反转链表段的头部、尾部以及前一个和后一个节点,提供了一种清晰且易于理解的解决方案。
LeetCode第92题反转链表 II
|
1月前
|
算法
LeetCode第21题合并两个有序链表
该文章介绍了 LeetCode 第 21 题合并两个有序链表的解法,通过创建新链表,依次比较两个链表的头节点值,将较小的值插入新链表,直至其中一个链表遍历完,再将另一个链表剩余部分接到新链表后面,实现合并。
LeetCode第21题合并两个有序链表
|
1月前
|
算法
LeetCode第19题删除链表的倒数第 N 个结点
该文章介绍了 LeetCode 第 19 题删除链表的倒数第 N 个结点的解法,通过使用快慢双指针,先将快指针移动 n 步,然后快慢指针一起遍历,直到快指针到达链尾,从而找到倒数第 N 个结点的前一个结点进行删除,同时总结了快慢指针可减少链表遍历次数的特点。
LeetCode第19题删除链表的倒数第 N 个结点
|
1月前
|
机器学习/深度学习
【刷题记录】相交链表
【刷题记录】相交链表
|
1月前
|
算法 Java
LeetCode初级算法题:环形链表+排列硬币+合并两个有序数组java解法
LeetCode初级算法题:环形链表+排列硬币+合并两个有序数组java解法
44 0