【数据结构】面试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 长于lenBstruct 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; }//到这里要么相等,要么链表不相交,已经走完了是NULLreturn longlist;
}

       

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何深度了解汤泉场所?VR全景给你答案

天气逐步转凉,温泉、水会这些室内汤泉场所开始登上消费的主战场。伴随着人们物质生活水平的提高,人们对休闲养生会馆的要求也愈发旺盛,汤泉场所也逐渐从单一的洗浴开始向休闲、娱乐、保健、桑拿等多种业态形式发展,那么大家如何深…

windiws docker 部署jar window部署docker 转载

Windows环境下从安装docker到部署前后端分离项目(springboot+vue) 一、前期准备 1.1所需工具: 1.2docker desktop 安装 二、部署springboot后端项目 2.1 部署流程 三、部署vue前端项目 3.1相关条件 3.2部署流程 四、前后端网络请求测试 一、前期准备 1.1所需工具: ①docke…

SQL自学三部曲_Part3:关于SQL必须要知道的一切

文章目录 一、关于SQL必须要知道的一切(1)SQL是什么?(2)你为什么要学SQL?(3)SQL到底要学什么? 二、 SQL学习的三个阶段(1)理解运行原理,串联核心语句(2&am…

AI换脸的一种技术实施例

刚刚看一个帖子的时候发现了AI识别中一个可以利用到其它场景的的一个通用处理步骤:人脸矫正。 人脸识别过程: 1.首先识别到关键的人脸部分,经过一个粗筛过程,把目标物的脸部图样先抓出来。 2.然后,因为人脸的水平&…

推介会如何做好媒体宣传

传媒如春雨,润物细无声,大家好,我是51媒体网胡老师。 推介会是一种专为企业、社会组织和团体、政府等提供的展示自身特点、产品和政策的活动形式,旨在促进交流活动,形成合作,从而带来共同利益。推介会的本…

C#WPF文本转语音实例

本文介绍C#WPF文本转语音实例 实现方法:使用类库(SpeechSynthesizer )实现的。 一、首先是安装程序包。 二、创建项目 需要添加引用using System.Speech.Synthesis; UI界面 <Windowx:Class="TextToSpeechDemo.MainWindow"xmlns="http://schemas.micr…

计算机组成原理-总线(学习这一篇就够了!)

目录​​​​​​​ 一、总线概述与结构 1.总线原理&#xff1a; 2.总线分类 &#xff08;1&#xff09;片内总线&#xff1a; &#xff08;2&#xff09;I/O总线&#xff1a; &#xff08;3&#xff09;系统总线&#xff1a; 3.总线的特性 &#xff08;1&#xff09;…

Angular 由一个bug说起之二:trackBy的一点注意事项

trackBy是angualr优化项目性能的一种方法, 通过返回一个具有绑定性的唯一值, 比如id&#xff0c;手机号&#xff0c;身份证号之类的&#xff0c;来让angular能够跟踪数组的项目&#xff0c;根据数据的变化来重新生成DOM, 这样就节约了性能。 但是如果是使用ngFor循环组件&…

如何下载 Apache + PHP + Mysql 集成安装环境并结合内网穿透工具实现公网访问内网服务

&#x1f308;个人主页&#xff1a;聆风吟 &#x1f525;系列专栏&#xff1a;网络奇遇记、Cpolar杂谈 &#x1f516;少年有梦不应止于心动&#xff0c;更要付诸行动。 文章目录 &#x1f4cb;前言一. WampServer下载安装二. WampServer启动三. 安装cpolar内网穿透3.1 注册账号…

2023年香港优才计划申请确实火爆,但请冷静!结合个人条件再考虑!

2023年香港优才计划申请确实火爆&#xff0c;但请冷静&#xff01;结合个人条件再考虑&#xff01; 自从香港优才计划实施取消年度限额&#xff0c;为期两年的优化措施以后&#xff0c;无论是申请数量还是获批数量都猛增。今年第二季度的获批量与第一季度的2,073宗相比&#xf…

RabbitMQ 安装及配置

前言 当你准备构建一个分布式系统、微服务架构或者需要处理大量异步消息的应用程序时&#xff0c;消息队列就成为了一个不可或缺的组件。而RabbitMQ作为一个功能强大的开源消息代理软件&#xff0c;提供了可靠的消息传递机制和灵活的集成能力&#xff0c;因此备受开发人员和系…

c# - - - Application.StartupPath(程序安装目录)和Environment.CurrentDirectory(程序工作目录)

Application.StartupPath 应用程序的安装目录&#xff0c;不会改变。 在C:\Users\Administrator\source\repos\ConsoleApp6\bin\Debug目录中&#xff0c;运行ConsoleApp6.exe。 安装目录为&#xff1a;C:\Users\Administrator\source\repos\ConsoleApp6\bin\Debug 在C:\Users…

​软考-高级-系统架构设计师教程(清华第2版)【第5章 软件工程基础知识(190~233)-思维导图】​

软考-高级-系统架构设计师教程&#xff08;清华第2版&#xff09;【第5章 软件工程基础知识&#xff08;190~233&#xff09;-思维导图】 课本里章节里所有蓝色字体的思维导图

050-第三代软件开发-软件部署脚本(二)

第三代软件开发-软件部署脚本(二) 文章目录 第三代软件开发-软件部署脚本(二)项目介绍软件部署脚本(二) 关键字&#xff1a; Qt、 Qml、 bash、 shell、 脚本 项目介绍 欢迎来到我们的 QML & C 项目&#xff01;这个项目结合了 QML&#xff08;Qt Meta-Object Languag…

外汇天眼:「外汇回撤」这术语是指什么?

"外汇回撤"是外汇交易中一个常见的专业术语。对于许多投资者&#xff0c;尤其是初学者来说&#xff0c;可能并不十分理解这个术语的实质。下面我们将详细介绍外汇回撤的含义。 外汇回撤通常被称为"外汇百分比回撤"。在外汇市场出现强烈趋势波动时&#xf…

Ubuntu中安装R语言环境并在jupyter kernel里面增加R kernel

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

文件上传 [SUCTF 2019]CheckIn1

打开题目 我们用cmd curl --head url 查看网站使用的是什么服务器 此题用的是openresty&#xff0c;OpenResty 是一个基于 Nginx 与 Lua 的高性能 Web 平台 我们上传php&#xff0c;phtml的一句话木马都显示不合法 那我们试试传a.jpg的一句话木马 显示我们一句话木马内容里面…

java面试常问

文章目录 java 基础1、JDK 和 JRE的区别2、 和equals的区别3、String、StringBuffer、StringBuilder4、String str “a”、 new String(“a”)一样吗&#xff1f;5、ArrayList 和 LinkedList的区别&#xff1f;6、HashMap的原理与实现6.1、容量与扩容6.2、扩容机制 7、HashMa…

MatrixOne完成与欧拉、麒麟信安的兼容互认

近日&#xff0c;超融合异构云原生数据库MatrixOne企业版软件V1.0完成了与欧拉开源操作系统&#xff08;openEuler简称“欧拉”&#xff09;、麒麟信安操作系统系列产品和虚拟化平台的相互兼容认证&#xff0c;通过了欧拉兼容性测评&#xff0c;获得了《openEuler技术测评证书》…

flutter背景图片设置

本地图片设置 1、在配置文件pubspec.yaml中&#xff0c;设置以下代码 assets:- assets/- assets/test/2、如果目录中没有assets文件夹&#xff0c;则创建一个文件夹&#xff0c;并且取名为assets&#xff0c;在此文件夹中存放图片资源即可&#xff0c;如果想分文件夹管理&…