【数据结构】链表专题3

ops/2024/9/25 17:20:36/

前言

本篇博客我们继续来讨论链表专题,今天的链表算法题是经典中的经典

💓 个人主页:小张同学zkf

⏩ 文章专栏:数据结构

若有问题 评论区见📝

🎉欢迎大家点赞👍收藏⭐文章

目录

1.判断链表是否有环

2.返回入环的第一个节点

3.随机链表的复制


1.判断链表是否有环

这道题链表尾指针很有可能指向链表中任何一个节点,所以是带环的意思,当然尾指针很有可能指向他自己

所以我们分析一下,该怎么判断带有环,有些人直接说我就判断是否和我原来的值相等,相等的话就是代表有环,但这种情况不能确保一定有环,因为即使我没进环也有可能值相等,所以这个行不通,所以我们要判断的话还得需要快慢指针,若快指针追上慢指针代表这链表有环,为什么快指针追上慢指针就会带有环那?

 我们来画图分析一下

slow和fast最初都在头结点, 我们让fast一次走两步,slow一次走一步,假如没换那么fast或fast->next就会指向空,假如有环,那么fast先进环,slow后进环,若fast追上slow就证明了这个链表就带有环

代码如下

这道题曾经被一个面试官提出新的问题

为什么一定会相遇,有没有可能会错过,永远追不上?

我们先来看第一个问题,我们假设slow进环后与fast距离为N, 环的长度是C

那么fast追击slow的过程距离变化如下:

N为偶数                 N为奇数 

N                            N

N-2                         N-2

N-4                         N-4

……                       ……

4                             3

2                            1

0                             -1

可以看出N若为偶数追上了,若N为奇数,则代表fast错过了,需要新的一轮追击,此刻他们之间的距离就变成了C-1,继续追击,我们根据第一轮追击可以得知,C-1是偶数的话代表第二轮追上了,C-1还是奇数的话,又错了一位,距离又变成了C-1,C-1既然是奇数,那就代表永远追不上了

所以追不上的条件前提是,第二轮的C-1是奇数,第一轮的N是奇数,但我们想想这两个条件会不会同时存在

这里我们就需要用到数学列等式的思维来判断两个条件是否可以同时存在

我们假设进环之前的距离是L

那么slow刚进环时,slow走过的距离是L,此刻我们假设fast走了x圈,那fast走过的距离就是L+x*C+C-N

fast的距离是slow的三倍

那么就有了等式

3L=L+x*C+C-N

换算为:2*L=(x+1)*C-N

偶数=(x+1)*偶数-奇数

我们根据数学运算法则中 ,N是奇数时,C必须是奇数,才能使等式成立,N是偶数时,C必须也是偶数,才能使等式成立。

所以,当N是奇数时,C为奇数,C-1为偶数,所以C-1不可能为奇数,所以不可能永远追不上,肯定相遇。                       

结论:一定能追上

N是偶数第一轮就追上了

N是奇数第一轮追不上,第二轮就追上了


2.返回入环的第一个节点

上面那道题是判断是否有环,这道题就是若有环,返回环的第一个节点,所以我们还是需要用到快慢指针,我们画图表示

如图,这里其实有个非常巧妙的方法,我们让慢指针一次走一步,快指针一次走两步,直到环里相遇,再创建两个指针,一个从头开始走,另一个从快慢指针相遇的地方开始走,俩指针一次走一步,这俩指针若相遇,则相遇的点必定是进环的首节点

代码如下

可是为什么那?

我们还是用数学的方法来证明一下 

我们假设,环之前的距离是L,环的长度为C,相遇点与入环点的距离为N,在慢指针进入环点时,快指针走了x圈

那么相遇时,slow走的距离是L+N

fast走的距离是L+x*C+N

fast走的路程是slow两倍

那么就有了等式

2*(L+N)=L+x*C+N

最后换算成L=(x-1)*C+C-N

假如x=1,那么L=C-N,正好是相遇点到入环首节点的距离与入环之前的距离相等,那么此时在头结点与相遇节点创建俩指针同时走,正好相遇在入环首节点,证实了我们上面的代码想法,但有人会想,你这里假设为1呀,我让它不为一,不唯一的话,相当于在相遇节点的指针多走了几圈C,最后还是在入环首节点相遇。


3.随机链表的复制

这道题链表每个节点里多了个指针指向随机节点,也有可能指向空,然后我们要深拷贝一份(深拷贝意思就是把指针指向对应的值对应关系也要在新拷贝的链表中实现),有人说我直接遍历然后拷贝不就行了,硬拷贝是可以的,但是有个问题,随机指针(random)指向的值如何在新链表中实现,有人说我在新链表里继续找就行呀,但是我们仔细想一下,我们链表里值有可能有时相等,所以如果你先拷贝过去,然后再去找对应的值,可能找到的值不是原链表对应的值,而是值相等的那个位置的节点。

比如就会出现图上这个情况,11找的就不是原来第四位的7,而是第一位的7,这就没拷贝成功。

所以这道题这种做法是不行的

我们先想一下,这道题我们要是先靠拷贝一下,然后插在原节点的后面其拷贝的节点就与源节点有了对应的关联关系

我们画图看一下

这一步代码如下

第二步控制random,拷贝完了,那拷贝那一份链表里的random怎么找那,其实很简单,拷贝的random是不是就是原值的random的next(这一点仔细想想,这一点想明白,这道题就没什么难点了)

第二步代码如下

第三步尾插新链表,将拷贝在原链表的节点尾插新链表,并返回新链表的头结点

代码如下

这道题整体代码如下

相当于三个while嘛,一个while循环一步


结束语 

链表有关算法题也就总结完了,从链表专题1到3都是特别经典的算法题,我们一定要反复练习掌握,OK,感谢观看!!!


http://www.ppmy.cn/ops/29290.html

相关文章

Redis基本命令

目录 一、包含String、Set数据类型的基本命令 1、添加一个键值对 2、获取key所关联的字符串值 3、同时设置多个key-value 4、获取多个key对应的值 运行结果 5、将给定的value追加到原值的末尾 追加后效果 6、删除单个key 7、同时删除多个key 8、查询包含某个字符的k…

MySql 主从同步-在原来同步基础上增加历史数据库

在MySql已经主从同步的后,由于有新的需求再增加1个历史数据库,要改原来的1个变成现在的2个数据库。在官网并没有找到类似的场景(官方同步多个数据是从一开始就设置,不是后续增加的),只能结合以往的经验自己…

【论文浅尝】Large Language Models for Generative Information Extraction: A Survey

本文对生成式IE的LLM进行了全面的探索。使用两种分类法对现有的代表性方法进行分类: (1)众多IE子任务的分类法,旨在对可以使用llm单独或统一提取的不同类型的信息进行分类; (2)学习范式分类法,对利用llm生成IE的各种新方法进行分类。 Preliminaries o…

【Python】使用 OpenCV 读取深度图,并转换深度图可视化显示

OpenCV 提供了多种工具来处理深度图数据,包括读取、显示、滤波、转换以及利用深度信息进行三维重建等。 1. 读取 使用IMREAD_UNCHANGED以保留深度信息 2. 转换 convertScaleAbs: 将深度图转换为更直观的表示形式,例如将其缩放到0-255范围以便显示。 …

java线上问题排查之磁盘和网络查看分析(二)

一、磁盘&IO df -lh 查看磁盘使用情况 Filesystem:文件系统 Size:容量 Used:已用 Avail:可用 Use%:已用百分比 Mounted on:挂载点 二、网络 查看TCP连接情况 常见问题 tcp队列溢出 netstat -s |e…

嵌入式前后台(Bare-Metal RTOS-Like)架构详解

前后台(Bare-Metal RTOS-Like)架构 在嵌入式系统开发中,针对资源有限的STM32微控制器,前后台(Bare-Metal RTOS-Like)架构是一种轻量级的实时性设计方法,它模拟了实时操作系统(RTOS&…

salesforce 如何访问lwc组件

访问lwc有哪些途径呢? Action ButtonTabAura use lwc(拓展)如何区分是新建页面还是编辑页面 Action Button xml文件中要配置tab<?xml version"1.0" encoding"UTF-8"?> <LightningComponentBundle xmlns"http://soap.sforce.com/2006/04/…

Vast+产品展厅 | Vastbase G100数据库是什么架构?(2)

Vastbase G100是海量数据融合了多年对各行业应用场景的深入理解&#xff0c;基于openGauss内核开发的企业级关系型数据库。 上一期&#xff0c;《Vast产品展厅》为您介绍了Vastbase G100的部署架构和物理架构。 本期&#xff0c;我们将为您详细讲解Vastbase G100的物理架构和…