链表之第三回

news/2024/5/9 10:49:24/文章来源:https://blog.csdn.net/m0_66780695/article/details/132380184

在这里插入图片描述

欢迎来到我的:世界

该文章收入栏目:链表

希望作者的文章对你有所帮助,有不足的地方还请指正,大家一起学习交流 !


目录

  • 前言
    • 第一题:判断是否为环形链表
    • 第二题:找到两条链表的相交点
    • 第三题:返回环形链表入环的那个结点
  • 总结

前言


今天的天气很不错😸,大早上起床,晨跑3公里,好久没这么跑了,真的很不错,我喜欢汗淋淋的感觉😄;暑假的假期时间也快到了,珍惜珍惜拥有自己所能支配的时间,来写几题吧✍️;


第一题:判断是否为环形链表


地址:oj地址


在这里插入图片描述
解题思路:

思路: 设置快慢指针,快指针 fast 每次走两个结点,慢指针 slow 每次走一个结点;如果是环形的,fast指针一定会和slow指针相遇,而且fast ,fast->next 走不到空(根据这两点做判断条件);如果不是环形,fast和slow一定不会相遇;可以根据这样的思路来写代码:
注意:
这个思路要想清楚

第一步

在这里插入图片描述

第二步:

在这里插入图片描述

第三步:相遇

在这里插入图片描述

代码实现:

bool hasCycle(struct ListNode *head) {struct ListNode *slow,*fast;//设置快慢指针slow=fast=head;//开始走,判断while(fast && fast->next){fast=fast->next->next; //快指针一次走两步slow=slow->next;	   //慢指针一次走一步	if(fast==slow)//相遇则代表是环形{return true;}}//对于环形单链表,不可能有指向空指针return false;
}

第二题:找到两条链表的相交点


地址:oj地址


在这里插入图片描述

解题思路:

思路: 首先要判断,既然他如果是相交的,那最起码是有最后一个节点是相交的吧,那就判断两条链表最后一个结点是不是一样的;要知道应该是从相交点开始后面的所有结点应该都相等,所以重点就在相交结点前;

在这里插入图片描述

创造两个变量count1,count2,计算两条链表的结点总数;创造两个指针cur1,cur2指向两条链表的头结点headA,headB,由cur1和cur2去找最后一个结点(这里注意:判断的条件应该是cur1所指向的next !=NULL,因为这里我们要找到最后一个结点),计算两条链表的各结点的数量count1,count2,先让长的链表头结点先走 | count1 - count2 | (绝对值可以用 abs这个函数 )步,这个时候就可以两条链表的从头结点开始比较,如果相等则返回这个结点,如果不相等,就走向下一个结点直到找到相交结点(这里一定会找到相交结点,因为在上面的时候已经将不相交的情况排除了);

第一步:计算出两条链表的结点数量

在这里插入图片描述

第二步: 让长的链表头结点先开始走 | count1 - count2 | 步,注意这里用到绝对值;

在这里插入图片描述

第三步开始进行比较判断,如果相等则返回这个结点,如果不相等,就走向下一个结点直到找到相交结点

在这里插入图片描述

注意:这里让长的链表先走,一般想法是用if语句;这里我想换个更方便的方法:创造两个指针变量fast,slow;分别指向两条链表的头结点,这里只要一个判断:如果count1 小于 count2 让fast指向headB,而slow指向headA;思想就是始终让fast指向的是长链表,slow指向短一点的链表;在之后就可以直接使用fast,slow变量了;

代码实现:

struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {int count1=1,count2=1;//计算两条链表的结点总数struct ListNode *cur1=headA;struct ListNode *cur2=headB;//找到headA链表的最后一个结点while(cur1->next){cur1=cur1->next;++count1;}//找到headB链表的最后一个结点while(cur2->next){cur2=cur2->next;++count2;}//结点不相等,则不可能相交;if(cur1!=cur2){return NULL;}  //绝对值int len =abs(count1-count2);//始终让fast指向长链表,slow指向短一点的链表;struct ListNode*fast=headA,*slow=headB;if(count1<count2){fast=headB;slow=headA;}//让其从相同位置结点开始比较;while(len--){fast=fast->next;}//比较while(fast != slow){fast=fast->next;slow=slow->next;}return fast;}

第三题:返回环形链表入环的那个结点


地址:oj地址


在这里插入图片描述
解题思路:

第一个简单的思路

设置两个指针fast和slow,fast一次走两步slow一次走一步,在环里的肯定会相遇,我们需要找到相遇的那个结点,然后断开这个结点与他后一个节点的链接,这样求入环点的问题就变成了:求两条链表的相交点;而求相交点的问题已经在上面解决了,是不是很简单理解;

在这里插入图片描述

将相遇点变成相交链表的尾巴,置成空;然后将相遇点tem下一结点设置成newnode指针,然后就是找相交点的问题了;

代码实现:

//实现找到相交结点并返回
struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {int count1=1,count2=1;//计算两条链表的结点总数struct ListNode *cur1=headA;struct ListNode *cur2=headB;//找到headA链表的最后一个结点while(cur1->next){cur1=cur1->next;++count1;}//找到headB链表的最后一个结点while(cur2->next){cur2=cur2->next;++count2;}//结点不相等,则不可能相交;if(cur1!=cur2){return NULL;}  //绝对值int len =abs(count1-count2);//始终让fast指向长链表,slow指向短一点的链表;struct ListNode*fast=headA,*slow=headB;if(count1<count2){fast=headB;slow=headA;}//让其从相同位置结点开始比较;while(len--){fast=fast->next;}//比较while(fast != slow){fast=fast->next;slow=slow->next;}return fast;
}struct ListNode *detectCycle(struct ListNode *head) {struct ListNode *fast=head,*slow=head;struct ListNode *tem=NULL,*newnode=NULL;while(fast && fast->next){fast=fast->next->next;slow=slow->next;//找到相遇点if(slow==fast){tem=slow;newnode=tem->next;//将相遇点变成链表的尾巴,置空;tem->next=NULL;//这边用我们之前写的求出相交结点return getIntersectionNode(head,newnode);}}return NULL;
}

这是一个思路比较容易理解的版本;还有一个比较难理解的思路:
是相似于物理的追击问题;
为来利用图解的方式;先进行这些假设:

在这里插入图片描述

这时候根据物理上的追击问题,根据fast和slow到相遇点所走的路程是一个两倍的关系;

在这里插入图片描述

所以:fast所走的距离是slow所走的距离的两倍;
slow走的距离是:L+X
fast走的距离是:L+X+n*C

  • 这里详解一下为什么fast走的距离是:L+X+n*C
    首先fast比slow走到快,当slow刚进入环,可能fast已经走了n圈环了(这个也是根据链表头结点到入环点的距离来判断的),且至少走了一圈了,当然不管他fast会绕多少圈,但他始终会在距离入环点 X的距离上的相遇点相遇的;

知道fast和slow所走的路程就可以用一个方程式:
2 * (L+X) = L+X+n * C (n>=1)化简得:
L = n * C - X
知道C - X 就是入口点;

在这里插入图片描述
无论fast走了多少圈C,C-X的距离就是相遇点到入环点的距离,无非是多绕了几圈,但是他们一定会在入环点相遇;

得出结论:一个指针从起点走,一个指针从相遇点走,他们一定会在入口点相遇;

代码的实现:

struct ListNode *detectCycle(struct ListNode *head) {struct ListNode *fast=head,*slow=head;//快慢指针struct ListNode *tem=NULL;//相交点//成环进while(fast && fast->next){fast=fast->next->next;slow=slow->next;//找到相遇点if(slow==fast){tem=slow;//开始找交点while(head!=tem){head=head->next;tem=tem->next;}return tem;}}//不成环就返回空return NULL;
}

总结


寻找环形链表的入环点的两种方法看似相同,其原理是不同的,看代码量也可以看出,前者是按照对环形的深刻理解方能很好的想到,而后者是根据其他知识求解出来的;如果还有更好的方法的老铁,可以在评论区里面一起进行讨论哦,在后面随着小孩的知识储备越多,小孩肯定还会加以优化优化!!


到了最后:感谢支持

我还想告诉你的是:
------------对过程全力以赴,对结果淡然处之
也是对我自己讲的

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

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

相关文章

P17~P18 电路定理 电路中难得的精彩到极致的电路理论

特勒根定理、互易定理、对偶定理比较难&#xff0c;非常重要&#xff0c;因为他们可以解决其他定理无法解决的问题。 1、特勒根定理1——个人感觉像能量守恒 特勒根定理与基尔霍夫定理齐名&#xff0c;与拓扑结构有关。都适用于任何线性非线性&#xff0c;时变的非时变的元件…

分类预测 | MATLAB实现S4VM半监督支持向量机二分类预测

分类预测 | MATLAB实现S4VM半监督支持向量机二分类预测 目录 分类预测 | MATLAB实现S4VM半监督支持向量机二分类预测分类效果基本介绍程序设计参考资料 分类效果 基本介绍 分类预测 | MATLAB实现S4VM半监督支持向量机二分类预测 程序设计 完整源码和数据获取方式&#xff1a; …

网站老域名跳转到新域名有哪些方法?内网穿透内网主机让外网访问

在网站服务器变更及本地主机搭建时&#xff0c;我们经常会遇到老域名地址跳转到新URL的配置&#xff0c;一些朋友还会面对无公网IP让外网访问的问题。今天我们来了解下网站老域名跳转到新域名有哪些方法&#xff0c;以及如何通过内网穿透实现内网主机让外网访问。 网站老域名跳…

Mac上传项目源代码到GitHub的修改更新

Mac上传项目源代码到GitHub的修改更新 最近在学习把代码上传到github&#xff0c;不得不说&#xff0c;真的还挺方便 这是一个关于怎样更新项目代码的教程。 首先&#xff0c;在本地终端命令行打开至项目文件下第一步&#xff1a;查看当前的git仓库状态&#xff0c;可以使用git…

线上异常的处理

一、线上问题的排查 进程ID 简称为PID free -m 查看内存使用情况 iostat 查看磁盘读写活动情况 netstat 查看网络连接情况 df -h 查看磁盘空间使用情况 du -sh 查看文件大小情况 1.1、top 命令查看CPU占用情况 top -n num 查看CPU占用最高的num个进程top -Hp PID 或 top -H -p…

【ROS】参数服务器--理论模型与参数操作(C++)

一、概念介绍 参数服务器在ROS中主要用于实现不同节点之间的数据共享。参数服务器相当于是独立于所有节点的一个公共容器&#xff0c;可以将数据存储在该容器中&#xff0c;被不同的节点调用&#xff0c;当然不同的节点也可以往其中存储数据。 作用&#xff1a;存储一些多节点…

【MySQL系列】SQL语句入门(创建删除操作)、字符集和数据类型详解

&#x1f490; &#x1f338; &#x1f337; &#x1f340; &#x1f339; &#x1f33b; &#x1f33a; &#x1f341; &#x1f343; &#x1f342; &#x1f33f; &#x1f344;&#x1f35d; &#x1f35b; &#x1f364; &#x1f4c3;个人主页 &#xff1a;阿然成长日记 …

HummingBird 基于 Go 开源超轻量级 IoT 物联网平台

蜂鸟&#xff08;HummingBird&#xff09; 是 Go 语言实现的超轻量级物联网开发平台&#xff0c;包含设备接入、产品管理、物模型、告警中心、规则引擎等丰富功能模块。系统采用GoLang编写&#xff0c;占用内存极低&#xff0c; 单物理机可实现百设备的连接。 在数据存储上&…

就算没有那个所谓的“国产保护月”,好莱坞电影也打不过中国电影

据路透社、美国文娱杂志《Variety》网站等18日报道&#xff0c;中国大陆暑期档票房在17日就已经超过了2019年同期的178亿元人民币&#xff0c;提前14天锁定了“史上最强暑期档”。这一份傲人的成绩单中&#xff0c;西方好莱坞电影所起到的作用却“微乎其微”。 更令人尴尬的是…

AI项目二:基于mediapipe的虚拟绘画

若该文为原创文章&#xff0c;转载请注明原文出处。 一、项目介绍 随着人工智能时代的到来&#xff0c;许多技术得到了空前的发展&#xff0c;让人们更加认识到了线上虚拟技术的强大。 通过mediapipe识别手的关键点&#xff0c;检测中指&#xff0c;实现隔空画画的操作。 通…

【LeetCode-中等题】128. 最长连续序列

题目 题解一&#xff1a;HeshSet枚举 思路&#xff1a;先对数组进行set去重&#xff0c;核心就是&#xff0c;先找出临界值&#xff08;假设以最小临界为例&#xff0c;那么这个临界值自己就是最小值&#xff0c;&#xff09;&#xff0c;以临界值不断做加1操作&#xff0c;看…

Excel/PowerPoint条形图改变顺序

条形图是从下往上排的&#xff0c;很多时候不是我们想要的效果 解决方案 选择坐标轴&#xff0c;双击&#xff0c;按下图顺序点击 效果

常见的 Python 错误及其解决方案

此文整理了一些常见的 Python 错误及其解决方案。 1、SyntaxError: invalid syntax 说明&#xff1a;无效的语法是最常见的错误之一&#xff0c;通常是由于编写代码时违反了 Python 的语法规则。可能的原因&#xff1a; 忘记在 if、while、for 等语句后写冒号&#xff0c;或者…

跟着NC学作图 | 使用python绘制折线图

写在前面 今天分享一篇使用Python绘制折线图的教程&#xff0c;在我们前提的教程中&#xff0c;关于使用R语言绘制折线图的教程也很少&#xff0c;跟着PC学作图 | 小提琴图Tufte箱形图折线图的绘制教程也只有相关一部分。 Python自己也是一直在学习&#xff0c;那么也就顺带分…

Python编程基础-函数

函数定义与调用 将完成某一特定功能并经常使用的代码编写成函数&#xff0c;在需要使用时直接调用 def 函数名(函数参数): 函数体 return 表达式或者值 def printHello(): #打印hello字符串print (hello)def printNum(): #输出0--9数字for i in range(0,10):print (i)return…

vue3 setup语法糖导入mixin

像这样直接导入&#xff0c;然后通过defineOptions声明mixin 然后就可以在这个组件使用mixin里的数据和方法了

java面试基础 -- ArrayList 和 LinkedList有什么区别, ArrayList和Vector呢?

目录 基本介绍 有什么不同?? ArrayList的扩容机制 ArrayLIst的基本使用 ArrayList和Vector 基本介绍 还记得我们的java集合框架吗, 我们来复习一下, 如图: 可以看出来 ArrayList和LinkedList 都是具体类, 他们都是接口List的实现类. 但是他们底层的逻辑是不同的, 相信…

攻防世界-warmup

原题解题思路 只有一张图片&#xff0c;就查看源代码&#xff0c;有一个source.php。 查看source.php&#xff0c;白名单中还有一个hint.php。 hint.php告诉我们flag的位置ffffllllaaaagggg 但是直接跳转是没用的&#xff0c;构造payload。 http://61.147.171.105:55725/sourc…

新版QQ NT 桌面版如何实现内存优化

一、背景 QQ 作为国民级应用,从互联网兴起就一直陪伴着大家,是很多用户刚接触互联网就开始使用的应用。而 QQ 桌面版最近一次技术架构升级还是在移动互联网兴起之前,在多年迭代过程中,QQ 桌面版也积累了不少技术债务,随着业务的发展和技术的进步,当前的架构已经无法很好…