【数据结构】面试OJ题——带环链表(数学推论)

 

目录

1.环形链表Ⅰ

​编辑

思路 :

思路拓展 

问题一:

问题二:

总结: 

问题三:

证明总结第三点 

 总结:

2. 环形链表Ⅱ

思路一

思路二 

3.相交链表 

 思路:


 

1.环形链表Ⅰ

141. 环形链表 - 力扣(LeetCode)

给你一个链表的头节点 head ,判断链表中是否有环。

如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。注意:pos 不作为参数进行传递 。仅仅是为了标识链表的实际情况。

如果链表中存在环 ,则返回 true 。 否则,返回 false 。

思路 :

        我们考虑环状时,如果是简单的循环next来判定,只会进入死循环;

需要引入新思维,判断是否带环,可以想象操场追赶问题,两个人在操场跑;让一个人先进场跑,再让另外一名选手进场,如果环足够大,如何让后入场的选手追赶上前面的;答案是,后来者加快速度。

        而在这道题中,我们用快慢指针概念,一个指针走两步,一个走一步,进入环后,快指针一定会和慢指针相遇

为什么一定会相遇呢?

距离为N,每次快指针走两步,慢指针走一步相当于距离每次减少1;

如此距离缩短,肯定会相遇,而当快慢指针相遇的时候,就是证明该链表带环;

而如果不带环,快指针将链表走完后,表明退出了循环即可; 

 

 

bool hasCycle(struct ListNode *head) {
    //快慢指针
    struct ListNode* slow = head;
    struct ListNode* fast = head;
    
    while(fast && fast->next)
    {
         fast = fast->next->next;
        slow = slow->next;
       if(fast == slow)
       {
           return true;
       }
    }
    return false;
}

思路拓展 

上面说到,距离缩小1一定会追上,那如果距离一次缩小2呢,是不是更快的会相遇呢

如果一次距离缩小n,是不是更快呢;如此延申,我们可以发现有三个拓展

 1. slow一次走一步,fast一次走两步,一定会相遇吗

2.slow一次走一步,fast一次走三步,一定会相遇吗

3.slow一次走n步,fast一次走m步,一定会相遇吗        (m>n>1)

问题一:

 距离缩小为1的话,一定会遇上,我们上面证明了

问题二:

 当slow进环时,fast和它的距离为N,一次减少2。分为两种情况

1.当N是偶数时:

2.当N是奇数时:

所以,我们可以发现,当N是偶数时,依然可以追上

当N是奇数是,得到下一轮:

这时候要看 C-1是奇偶数

如果c-1是偶数:在下一轮就会追上,

如果c-是奇数:则会开始无限循环下去,一直追击不上;

总结: 

1.如果N是偶数,第一轮内就会追上;

2.如果N是奇数,C是奇数,第一轮会错过,下一轮会追击(C是奇数,C-1就会是偶数);

3.如果N是奇数,C是偶数,就永远追不上;

问题三:

 这个类似问题二,每次距离缩小是m-n(>=1);

如果N%(m-n) == 0  说明一圈就可以追上;

如果是-1,则是C-1,-2就是C-2;

如果这个数是m-n的倍数,如果不是倍数,则可能存在死循环        

证明总结第三点 

写到这里时,问题二的 总结第三点,是否真的成立呢?

3.如果N是奇数,C是偶数,就永远追不上;

 此刻,slow走了L的路程,fast走了 L + nC - N; 

在这时,可以推论出公式 

 

 推论出来的这个公式:

2L:一定是偶数,

n*C:总结第三点假设这里是偶数;

N:如果要满足此公式,N就不可能会奇数,

所以, 3.如果N是奇数,C是偶数,就永远追不上;

这句话的条件不成立

 总结:

1.如果N是偶数,第一轮内就会追上;

2.如果N是奇数,C是奇数,第一轮会错过,下一轮会追击(C是奇数,C-1就会是偶数);

只存在这两种

bool hasCycle(struct ListNode *head) {
    //快慢指针
    struct ListNode* slow = head;
    struct ListNode* fast = head;
    
    while(fast && fast->next)
    {
         fast = fast->next->next;
        slow = slow->next;
       if(fast == slow)
       {
           return true;
       }
    }
    return false;
}

 

在这种情况下,该情况其实还可以推论出另一个公式结论 。这就是第二题

2. 环形链表Ⅱ

 142. 环形链表 II - 力扣(LeetCode)

给定一个链表的头节点  head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null

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

不允许修改 链表。

思路一

 我们依旧用上一道题的画图思路来解决,

第一步先判断是否存在环,

第二部找到入口点;

 

用快慢指针方法,slow走一步,fast走两步

 

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(fast == slow) //相遇了
       {
        struct ListNode* meet = slow;
          while(head != meet)   //起点和相遇点
          {
              head = head->next;
              meet = meet->next;
          }
          //找到入口点了
          return meet;
       }
    }
    return false;
}

思路二 

 这道题还有另外一种解法,就是在相遇点meet这里,将下一个节设为newnode,再将meet->next置NULL,这样就将带环链表问题转换成 相交链表问题

而这种解法自然有相交链表的题;

但是这种解法比较复杂,建议用思路一的

3.相交链表 

160. 相交链表 - 力扣(LeetCode)

给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回 null 。

题目数据 保证 整个链式结构中不存在环。

注意,函数返回结果后,链表必须 保持其原始结构 。

 

 思路:

题目两种情况,存在相交和不相交,

相交还有前后长短问题;

1.先判断是否有相交点;如果相交,尾节点一定相同;

2.计算两条链表的长度,让长链表先走差距步,这样可以让两条链表处于相同head

3.再让两条链接一起走,直到遇到相同点;

返回相遇点或者是结束链表遍历返回NULL

 

struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {
    struct ListNode* curA = headA,*curB = headB;
    //计算两条链表长度
    int lenA = 0;
    int lenB = 0;
    while(curA)
    {
        lenA++;
        curA = curA->next;
    }
    while(curB)
    {
        lenB++;
        curB = curB->next;
    }

    //算出长度差
    int n = abs(lenA-lenB);
    //假设lenA 长于lenB
    struct ListNode* longlist = headA,*shortlist = headB;
    if(lenB>lenA) //假设不成立,交换
    {
        longlist = headB;
        shortlist = headA;
    }
    //长的先走差值
    while(n--)
    {
        longlist = longlist->next;
    }
    //判断是否相等
    while(longlist != shortlist)
    {
        longlist = longlist->next;
        shortlist = shortlist->next; 
    }
    //到这里要么相等,要么链表不相交,已经走完了是NULL
    return longlist;
}

       

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/144368.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

AI生成图片教程(基于DALL-E3)

目录 前言new bingImage Creator 前言 今天登录GPT时发现openai的官网显示其有生成图片的模型DALL-E3,于是想试一试其效果如何。 奈何ChatGPT只能在付费版上使用,但是这个DALL-E3其实免费使用。 使用途径有两个: new bing 进入new bing 很…

好心提醒下,幼师姐妹们要知道啊

幼师家人们在不在?在不在? 不会还有姐妹在自己写教案,写总结,写评语啥的吧,这个好东西真的要知道啊!! 只要输入关键词,马上就能得到你想要的内容,真的很强啊&#xff0…

sinc 函数

See https://wuli.wiki/online/sinc.html 公式(3)的证明见 https://wuli.wiki/online/JdLem.html#ex_JdLem_1 百度百科

5G车载网关让医院无人配送车“灵活“起来

​ 5G车载网关应用于无人医院配送车 随着社会老龄化加剧和医疗需求增长,为患者提供及时、便捷的药品配送服务成为医院的一项重要任务。传统的人工配送方式效率低下,无法满足患者的实时配送需求。针对这一痛点,5G车载网关为无人医院配送车提供了有力的技术支撑。 5G车载网关集成…

ubuntu小技巧30--23.10桌面版安装钉钉启动报错undefined symbol: FT_Get_Color_Glyph_Layer

ubuntu小技巧30-- 23.10桌面版安装钉钉启动报错undefined symbol: FT_Get_Color_Glyph_Layer 介绍解決方法说明 介绍 近期在电脑上安装了 ubuntu 23.10桌面版本, 安装最新版钉钉后无法正常打开软件,报错 undefined symbol: FT_Get_Color_Glyph_Layer ,具…

安装部署PowerDNS--实现内网DNS解析(use)

使用PowerDNS实现内网DNS解析_powerdns-admin-CSDN博客 https://www.cnblogs.com/guangdelw/p/17348982.html 一、概念介绍 PowerDNS是一个域名解析服务,官网提供了三个组件:Authoritative、Recursor、dnsdist,分别用来作为权威服务器、域名递…

C++ VS2015安装教程,下载和安装(下载地址+图解+详细步骤)

说明:VS2015的三个版本分别为: Visual Studio Community(社区版):满足大部分程序员的需求(推荐) Visual Studio Professional(专业版) Visual Studio Enterprise(企业版) 1、下载地址(这里只提供Community版) htt…

5 个基本步骤,学会创建自己的CRM流程

客户关系管理 (CRM) 系统是必备的客户数据库工具,用于跟踪潜在客户、现有客户、接触点等。使用可靠的技术来跟踪所有客户数据固然重要,但为该技术制定的流程更为关键。因这是用来管理客户和整个客户生命周期的策略。 什么是CRM流程? CRM流程…

如何利用「深度上下文兴趣网络」提升点击率?

美团到店广告平台在用户行为序列建模算法的迭代落地中,基于对业务实际场景中用户决策心智的观察,创新性地提出了深度上下文兴趣网络,精确建模了用户的兴趣,提升了CTR等线上业务指标。本文介绍了相应算法背后的动机、建模方法以及工…

高防CDN:构筑网络安全的钢铁长城

在当今数字化的世界里,网络安全问题日益突显,而高防CDN(高防御内容分发网络)正如一座坚不可摧的钢铁长城,成为互联网安全的不可或缺之物。本文将深入剖析高防CDN在网络安全环境中的关键作用,探讨其如何构筑…

漏洞复现--用友U8-cloud RegisterServlet SQL注入

免责声明: 文章中涉及的漏洞均已修复,敏感信息均已做打码处理,文章仅做经验分享用途,切勿当真,未授权的攻击属于非法行为!文章中敏感信息均已做多层打马处理。传播、利用本文章所提供的信息而造成的任何直…

提高软件测试覆盖率的5个重点

软件测试覆盖率是软件测试中的一个重要指标,它有利于保障软件质量、提高软件可靠性和可维护性。软件测试覆盖率能够发现并修复代码缺陷,确保代码的正确性,提高软件的稳定性,降低成本和风险。 因此进一步提高软件测试覆盖率对于软件…

pip 更换国内镜像

方法 1 在C:\Users\85249\AppData\Roaming位置新建一个pip文件夹(之前已经有了就不用管) 在文件夹里面新建一个pip.ini文件。 文件一开始是空的,用文本文档打开后写入如下文所示。这里使用的是清华源,而且设置为信任&#xff0…

各路大神献出自定义GPTs,24小时Top名单

没有 GPTs 做不到的,只有你想不到的。 11 月 10 日凌晨, OpenAI 上线 GPTs,所有的 ChatGPT Plus 订阅用户都可以自己定制 GPT,无需任何编码知识,在聊天过程中就构建好了。 发布当天,OpenAI CEO 山姆・奥特曼…

若依框架修改包名报错

1.首先看下报错截图 启动GateWay 2.这个是因为 我改了里面的包名就是下面 ruoyi改成screen爆了上面的问题 3.那么关键的来了,我测了下 改了core不管启动gateway还是modules里面任何一个都会爆打不开工具类的问题 ,我看了其他pom也没有引用core&#xff…

超详细!必看!!STM32--时钟树原理

一、什么是时钟? 时钟是单片机的脉搏,是系统工作的同步节拍。单片机上至CPU,下至总线外设,它们工作时序的配合,都需要一个同步的时钟信号来统一指挥。时钟信号是周期性的脉冲信号。 二、什么是时钟树? S…

K8S的基础知识

K8S的意义与入门 专有名词 容器:包含了运行一个应用程序所需要的所有东西,包括:代码、运行时、各种依赖和配置。pod:K8s调度的最小单元,包含一个或多个容器。一个容器组中的容器具有紧密耦合性,共享资源,存储空间和IP。即同一个容器组中的容器可以通过localhost:xxx访问…

中馥集团双11当日发货销售额突破1000万!

昨日,中馥集团双十一当日发货销售额突破1000万,再创新高!双十一大促期间,中馥集团全体上下通力合作,每场直播商品经层层筛选、严格评选的“名品”,既有优质精品文化酒,也有市场火爆的高端酱酒&a…

【JavaEE初阶】IP协议简介

文章目录 前言🌴IP协议的概念🌳IP数据报🚩IPv4协议头格式🚩IPv6的诞生 🎍IP地址🚩IP地址的格式:🚩IP地址的分类🎈网络号与主机号的划分 🚩特殊的IP地址&#…

stable diffusion comfyui的api使用教程

一、为什么要使用comfyui的api?对比webui的api,它有什么好处? 1、自带队列 2、支持websocket 3、无需关心插件是否有开放api接口,只要插件在浏览器中可以正常使用,接口就一定可以使用 4、开发人员只需关心绘图流程的搭建 5、切换…