LeetCode 142.环形链表2 C写法

LeetCOde 142.环形链表2 C写法

思路1🤔:

​ 用环形链表的方法,快慢指针找到 slow和fast的相遇点 ,此时头到入口点的位置相遇点到入口点距离一样

​ 我们假设头到入口点的长度为L,环的长度为C,相遇点到入口点的长度为X。那么我们可以计算出在相遇时,fast走了L+n*C+X 步(其中n为fast走的圈数),slow走了L+X 步,且fast步数是slow的两倍 ,进一步得出等式 2(L+X) = L+n*C+X ,化简后得到L = n*C -X,然而n无论走多少圈,最后还是在相遇点,所以可以把n看做1,最后得L = C - XC-X就是相遇点到入口点的长度,等式成立,推出该方法。

代码🔎:

c 复制代码
struct ListNode *detectCycle(struct ListNode *head) {
    struct ListNode* slow = head;
    struct ListNode* fast = head;
    while(fast && fast->next)
    {
        fast = fast->next->next;
        slow = slow->next;
        if(slow == fast) //带环则开始找入口点
        {
            struct ListNode* meet = slow;
            while(meet != head) //meet等于head时代表找到入口点
            {
                meet = meet->next;
                head = head->next;
            }
            return meet;
        }
    }
    return NULL;
}

思路2🤔:

​ 在相遇点用一个指针meet等于slow,meet再找到meet的next,然后将slow->next指向空,那么下一次循环的时候就会在空结点停下 ,将环形链表变为相交链表

代码🔎:

c 复制代码
struct ListNode *detectCycle(struct ListNode *head) {
    struct ListNode* slow = head;
    struct ListNode* fast = head;
    while(fast && fast->next)
    {
        fast = fast->next->next;
        slow = slow->next;
        if(slow == fast)
        {
            struct ListNode* meet = slow;
            meet = meet->next; 
            slow->next = NULL; //断环
            struct ListNode* List1 = head;
            struct ListNode* List2 = meet;
            int LenHead = 1;
            int LenMeet = 1;
            int gap = 0;
            while(List1->next) //计算走到尾需要多少步
            {
                ++LenHead;
                List1 = List1->next;
            }
            while(List2->next)
            {
                ++LenMeet;
                List2 = List2->next;
            }
            if(List1 == List2) //这里可以不写,因为有相遇点必定是相交链表
            {
                struct ListNode* shortList = head; //先假设头结点到入口点的长度更短
                struct ListNode* longList = meet;
                gap = abs(LenHead-LenMeet); //更长的的先走gap步
                if(LenHead > LenMeet) //如果假设错了就交换
                {
                    shortList = meet;
                    longList = head;
                }
                while(gap--) //让长短链表距离相等
                {
                    longList = longList->next;
                }
                while(longList != shortList) //相等就说明找到入口点
                {
                    longList = longList->next;
                    shortList = shortList->next;
                }
                return longList;
            }
        }
    }
    return NULL;
}
相关推荐
是苏浙1 小时前
零基础入门C语言之枚举和联合体
c语言·开发语言
熬了夜的程序员1 小时前
【LeetCode】99. 恢复二叉搜索树
算法·leetcode·职场和发展
Kent_J_Truman2 小时前
LeetCode Hot100 自用
算法·leetcode·职场和发展
还是码字踏实2 小时前
算法题种类与解题思路全面指南:基于LeetCode Hot 100与牛客Top 101
算法·leetcode
ACP广源盛139246256733 小时前
(ACP广源盛)GSV2231---DisplayPort 1.4 MST 到 HDMI 2.0/DP/Type-C 转换器(带嵌入式 MCU)
c语言·开发语言·单片机·嵌入式硬件·音视频·mst
QT 小鲜肉3 小时前
【QT/C++】Qt网络编程进阶:UDP通信和HTTP请求的基本原理和实际应用(超详细)
c语言·网络·c++·笔记·qt·http·udp
星释3 小时前
Rust 练习册 8:链表实现与所有权管理
开发语言·链表·rust
Yurko134 小时前
【C语言】选择结构和循环结构的进阶
c语言·开发语言·学习
熬了夜的程序员4 小时前
【LeetCode】101. 对称二叉树
算法·leetcode·链表·职场和发展·矩阵
范纹杉想快点毕业4 小时前
12个月嵌入式进阶计划ZYNQ 系列芯片嵌入式与硬件系统知识学习全计划(基于国内视频资源)
c语言·arm开发·单片机·嵌入式硬件·学习·fpga开发·音视频