【代码随想录 | Leetcode | 第七天】链表 | 链表相交 | 环形链表 II

前言

欢迎来到小KLeetcode|代码随想录|专题化专栏,今天将为大家带来链表相交和环形链表 II的分享

目录

  • 前言
  • [面试题 02.07. 链表相交](#面试题 02.07. 链表相交)
  • [142. 环形链表 II](#142. 环形链表 II)
  • 总结

面试题 02.07. 链表相交

题目链接点这里

给你两个单链表的头节点 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

如果 listAlistB 有交点,intersectVal == listA[skipA + 1] == listB[skipB + 1]
思路:这道题目也是双指针的一种用法,我们先定义两个指针分别指向两个链表的头结点,然后让指向较长链表的指针朝后移动,和较短链表的那个指针对其,最后就是移动,循环比较,返回第一个 两个指针指向同一节点的情况,没找到则返回空

cpp 复制代码
class Solution {
public:
    ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {
        ListNode* curA = headA, *curB = headB;
        int lenA = 0, lenB = 0;
        while (curA != nullptr) {
            lenA++;
            curA = curA->next;
        }
        while (curB != nullptr) {
            lenB++;
            curB = curB->next;
        }
        curA = headA;
        curB = headB;
        if (lenA < lenB){
            swap(lenA,lenB);
            swap(curA,curB);
        }
        int gap = lenA - lenB;
        while (gap--) curA = curA->next;
        while (curA != nullptr) {
            if(curA == curB) return curA;
            curA = curA->next;
            curB = curB->next;
        }
        return nullptr;
    }
};

142. 环形链表 II

题目链接点这里
给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。注意:pos 不作为参数进行传递,仅仅是为了标识链表的实际情况。
不允许修改 链表。

示例 1:

输入 :head = [3,2,0,-4], pos = 1
输出 :返回索引为 1 的链表节点
解释:链表中有一个环,其尾部连接到第二个节点。

示例 2:

输入 :head = [1,2], pos = 0
输出 :返回索引为 0 的链表节点
解释:链表中有一个环,其尾部连接到第一个节点。

示例 3:

输入 :head = [1], pos = -1
输出 :返回 null
解释:链表中没有环。

提示:
链表中节点的数目范围在范围 [0, 104] 内

-105 <= Node.val <= 105
pos 的值为 -1 或者链表中的一个有效索引

思路:解决这个问题我们需要明确下面两个问题

  • 判断链表是否有环
  • 如果有环,如何找到这个环的入口

问题一:判断链表是否有环

可以使用快慢指针法,定义一个快指针 fast 和一个慢指针 slow ,从头结点开始,每次让快指针移动两个节点,慢指针每次移动一个节点,如果在途中相遇,则代表有环

相遇一定在环内相遇,这是为什么?快指针一定比慢指针先入环

为什么快慢指针一定会相遇?快指针一次移动两步,而慢指针一次移动一步,所以快指针相当于是以每次一步来追赶慢指针,所以一定会相遇

问题二:找环的入口

假设从头结点到环形入口节点 的节点数为x。 环形入口节点到 fast指针与slow指针相遇节点 节点数为y。 从相遇节点 再到环形入口节点节点数为 z。

那么相遇时: slow指针走过的节点数为: x + y, fast指针走过的节点数:x + y + n (y + z),n为fast指针在环内走了n圈才遇到slow指针, (y+z)为 一圈内节点的个数A
则有:(x + y) * 2 = x + y + n (y + z)
化简得x = (n - 1)(y + z) + z ,注意这里n一定是大于等于1的,因为 fast指针至少要多走一圈才能相遇slow指针。 当n等于一和大于一的效果是一样的,因为当n大于一就相当于快指针在环内多转了几圈

cpp 复制代码
class Solution {
public:
    ListNode *detectCycle(ListNode *head) {
        ListNode* slow = head;
        ListNode* fast = head;
        while (fast != nullptr && fast->next != nullptr) {
            slow = slow->next;
            fast = fast->next->next;
            if (slow == fast) {
                ListNode* index1 = head;
                ListNode* index2 = fast;
                while (index1 != index2) {
                    index1 = index1->next;
                    index2 = index2->next;
                }
                return index1;
            }
        }
        return nullptr;
    }
};

总结

今天的收获还是挺大的,这个环形链表不难,但是很有意思,有点写数学题的感觉,一步一步揭开面纱真的很舒服~

相关推荐
好吃的肘子41 分钟前
Elasticsearch架构原理
开发语言·算法·elasticsearch·架构·jenkins
胡耀超1 小时前
霍夫圆变换全面解析(OpenCV)
人工智能·python·opencv·算法·计算机视觉·数据挖掘·数据安全
软行1 小时前
LeetCode 每日一题 3341. 到达最后一个房间的最少时间 I + II
数据结构·c++·算法·leetcode·职场和发展
nlog3n1 小时前
Go语言交替打印问题及多种实现方法
开发语言·算法·golang
How_doyou_do1 小时前
备战菊厂笔试4
python·算法·leetcode
朱剑君1 小时前
第九天——贪心算法——非递减数组
算法·贪心算法
Wnq100722 小时前
工业场景轮式巡检机器人纯视觉识别导航的优势剖析与前景展望
人工智能·算法·计算机视觉·激光雷达·视觉导航·人形机器人·巡检机器人
天上路人4 小时前
AI神经网络降噪算法在语音通话产品中的应用优势与前景分析
深度学习·神经网络·算法·硬件架构·音视频·实时音视频
好吃的肘子4 小时前
MongoDB 应用实战
大数据·开发语言·数据库·算法·mongodb·全文检索
汉克老师4 小时前
GESP2025年3月认证C++二级( 第三部分编程题(1)等差矩阵)
c++·算法·矩阵·gesp二级·gesp2级