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;
}



相关文章
【力扣】-- 移除链表元素
【力扣】-- 移除链表元素
163 1
Leetcode第21题(合并两个有序链表)
这篇文章介绍了如何使用非递归和递归方法解决LeetCode第21题,即合并两个有序链表的问题。
204 0
Leetcode第21题(合并两个有序链表)
|
6月前
|
机器学习/深度学习 算法
24. 两两交换链表中的节点, 19.删除链表的倒数第N个节点 ,面试题 02.07. 链表相交
1. **两两交换链表中的节点**:通过引入虚拟头结点,使所有节点都能采用统一的交换逻辑,避免对头结点单独处理。 2. **删除链表的倒数第N个节点**:利用双指针技巧,让快慢指针保持N个节点的距离,当快指针到达末尾时,慢指针正好指向待删除节点的前一个节点。 3. **链表相交**:先计算两链表长度并调整起点,确保从相同距离末尾的位置开始遍历,从而高效找到相交节点或确定无交点。 以上方法均在时间复杂度和空间复杂度上进行了优化,适合用于理解和掌握链表的基本操作及常见算法设计思路。
|
8月前
|
算法 Go
【LeetCode 热题100】23:合并 K 个升序链表(详细解析)(Go语言版)
本文详细解析了 LeetCode 热题 23——合并 K 个升序链表的两种解法:优先队列(最小堆)和分治合并。题目要求将多个已排序链表合并为一个升序链表。最小堆方法通过维护节点优先级快速选择最小值,;分治合并则采用归并思想两两合并链表。文章提供了 Go 语言实现代码,并对比分析两种方法的适用场景,帮助读者深入理解链表操作与算法设计。
310 10
|
算法
❤️算法笔记❤️-(每日一刷-160、相交链表)
❤️算法笔记❤️-(每日一刷-160、相交链表)
142 1
LeetCode第二十四题(两两交换链表中的节点)
这篇文章介绍了LeetCode第24题的解法,即如何通过使用三个指针(preNode, curNode, curNextNode)来两两交换链表中的节点,并提供了详细的代码实现。
154 0
LeetCode第二十四题(两两交换链表中的节点)
Leetcode第十九题(删除链表的倒数第N个节点)
LeetCode第19题要求删除链表的倒数第N个节点,可以通过快慢指针法在一次遍历中实现。
202 0
Leetcode第十九题(删除链表的倒数第N个节点)
|
索引
力扣(LeetCode)数据结构练习题(3)------链表
力扣(LeetCode)数据结构练习题(3)------链表
275 0
【LeetCode 10】142. 环形链表 II
【LeetCode 10】142. 环形链表 II
115 0
【LeetCode 09】19 删除链表的倒数第 N 个结点
【LeetCode 09】19 删除链表的倒数第 N 个结点
198 0