环形链表 [两道题目](详解)

环形链表(详解)

第一题:

题目:

题目链接

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

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

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

示例 1:

img

输入:head = [3,2,0,-4], pos = 1
输出:true
解释:链表中有一个环,其尾部连接到第二个节点。

示例 2:

img

输入:head = [1,2], pos = 0
输出:true
解释:链表中有一个环,其尾部连接到第一个节点。

示例 3:

img

输入:head = [1], pos = -1
输出:false
解释:链表中没有环。

思路:

首先我们创建一个快慢指针,让其在链表内不断走,如果链表是带环的,那么快指针迟早会追上慢指针,追上了那就是带环链表。

但是注意了在这里

我们的快慢指针之间的速度差距一定得是——fast = 2slow,即是慢指针走1步,快指针走两步。

为什么呢?

我们假设slow指针进入环时离fast指针的距离是X。

  1. 当X == 0的时候,slow一进环就碰到fast了,说明第一个节点到环的长度和环内的长度是一样的。
  2. 如果slow走到环的长度是,环内的一半长度,这个时候,slow进环的时候,fast刚好在环中间。如图所示

image-20240503162211058

  1. 也就是说,当slow慢指针入环之后,fast和slow每走一次,他们之间的距离X都会-1.如图所示

image-20240503162320737

其实就证明了,slow入环之后,和fast的距离是X,slow再走X次,fast就能追上slow指针。

那为什么说fast只能是slow的两倍速率呢?

因为如果我们的fast走其他步 比如3步 4步,就有可能会不相遇!

image-20240503163459651

如图所示,如果X是奇数,比如是1,那么一次走三步就会直接跳过去,那么此时的X就会重置为环的长度-1,如果这个新的X也是奇数,那么就会死循环!

fast走其他步也是同样的道理,这里就不在例举。

因此其他步总有一些特殊的情况无法处理,无法相遇。

但是如果fast每次走两步,每次的距离减少1 那就一定会相遇。

因此这里只能让fast = 2slow。

慢指针在环内是走不到一圈的,严格来说,最差的情况是slow在入环第一个节点,fast在入环第二个节点,这个时候慢指针走接近2/3圈就会被追上。

代码实现:

struct ListNode 
{
    int val;
    struct ListNode* next;
};

typedef struct ListNode ListNode;
bool hasCycle(struct ListNode* head)
{
    // 创建快慢指针
    ListNode* slow, * fast;
    slow = fast = head;

    // 快慢指针都在环里面一直走下去,快指针迟早能追上慢的指针
    while (fast && fast->next) // 我们不知道传进来的链表是不是带环的因此,fast和fast->next 都不能为NULL
    {
        slow = slow->next;
        fast = fast->next->next; // 注意了这里的快指针只能是 fast = 2slow 
        // 如果快指针追上慢指针那就是带环
        if (fast == slow)
            return true;
    }
    // 走到这里说明 不带环 
    return false;
}

第二题:(难度:中等)

题目:

题目链接

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

不允许修改 链表。

示例 1:

img

输入:head = [3,2,0,-4], pos = 1
输出:返回索引为 1 的链表节点
解释:链表中有一个环,其尾部连接到第二个节点。

示例 2:

img

输入:head = [1,2], pos = 0
输出:返回索引为 0 的链表节点
解释:链表中有一个环,其尾部连接到第一个节点。

示例 3:

img

输入:head = [1], pos = -1
输出:返回 null
解释:链表中没有环。

思路:

思路1:(理解容易,写起来较麻烦)
  1. 创建快慢指针,如果有环就相遇,无环无法相遇
  2. 首先让fast和slow相遇,找到这个相遇的节点。
  3. 然后把相遇的节点断开
  4. 然后相遇点和到相遇点的上一个节点是一个链表,一开始的链表头到断点也是一个链表。
  5. 这个时候我们把问题转化成了两个链表的相交问题,找到交点,这个交点就是题目要找的入环的第一个节点!

image-20240503182909788

代码实现:
struct ListNode 
{
    int val;
    struct ListNode* next;
};

typedef struct ListNode ListNode;
struct ListNode *detectCycle(struct ListNode *head) 
{
    // 排除空链表的情况
    if(head == NULL)
        return NULL;

    // 创建快慢指针
    ListNode* slow, *fast;
    slow = fast = head;

    // 让快慢指针遍历链表,看看带环
    while(fast && fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
        // 判断是否有相遇点
        if(slow == fast)
            break;
    }

    // 排除不带环的情况
    if(slow != fast)
        return NULL;

    // 让相遇点断开 
    ListNode* next = slow->next; // 让next存储相遇点的下一个节点
    slow->next = NULL; // 断开相遇点

    // 这个时候next 和 head 分别为两条链表
    // 这个时候我们就找这两个链表的相交点就好了,交点就是入环第一个节点

    // 要想找到交点首先要求两个链表的长度
    ListNode* curhead = head;
    ListNode* curnext = next;
    int lenhead = 0;
    int lennext = 0;

    // 遍历head链表,统计长度
    while(curhead)
    {
        lenhead++;
        curhead = curhead->next;
    }

    // 统计next链表长度
    while(curnext)
    {
        lennext++;
        curnext = curnext->next;
    }


    // 默认head为起始点的为长链表 next为起始点的为短链表
    ListNode* longlist = head;
    ListNode* shortlist = next;

    // 判断两个链表的长度是否和我们默认的情况相同
    if(lenhead < lennext)
    {   
        // 不同就改过来
        longlist = next;
        shortlist = head;
    }

    // 计算两个链表的长度差
    int gap = abs(lenhead - lennext);
    // 让长链表走两个链表的长度差
    while(gap--)
    {
        longlist = longlist->next;
    }


    // 这个时候长短链表的长度相同了
    while(longlist)
    {
        // 这个时候要判断next指向的是否是入环的第一个节点,因为快慢指针的相遇点可能是入环的最后一个节点
        // 因此要先判断是否为交点
        if(longlist == shortlist)
            return longlist;
        // 让两个链表都向后走
        longlist = longlist->next;
        shortlist = shortlist->next;
    }
    
    // 如果走到这里就说明有情况没有排除
    // 比如链表只有一个节点且不带环
    return NULL;
}
思路2:(理解较难,写起来比较容易)

注意 思路2一定要知道推论是怎么推出来的,要理解透彻代码是怎么来的!

  1. 首先找通过快慢指针找到相遇点。

  2. 我们假设第一个节点到入环的第一个节点的距离叫做L,把slow指针入环后走的距离叫X,环的长度是C。如图所示

  3. image-20240503200031822

  4. 那么慢指针走的距离就是:L + X (慢指针在环内一定走不过一圈)

  5. 快指针走的距离就是: L + N * C + X (N代表走的环的圈数)

如果L短C大 也就是环的长度大 那么N就小,就是快指针绕环走的圈数少

如果L长C小,也就是环的长度小,N就大,就是快指针要绕环走很多圈,慢指针才能入环。

image-20240503200737642

如图所示,L长C小的情况,N会比较大

  1. 我们又知道 2slow = fast 也就是2(L + X) = L + N*C + X
  2. 推论得知 L = N*C - X. 也就是第一个节点到入环的第一个节点的距离L = 快指针在环内走的长度N * C 再减去一个慢指针入环的距离 X

也可以理解成 L = (N-1) C + C - X*

也就是fast走了(N-1)* C 的距离后再走了一个 C - X的距离就是L的长度

image-20240503201618422

  1. 也就是说从相遇点有一个指针,和链表的起始点开始一起走,两个指针相遇的节点就是我们要找的入环的第一个节点
代码实现:
struct ListNode 
{
    int val;
    struct ListNode* next;
};

typedef struct ListNode ListNode;
struct ListNode *detectCycle(struct ListNode *head) 
{
    // 首先创建快慢指针
    ListNode* slow, *fast;
    slow = fast = head;

    // 通过快慢指针判断是否带环,并且让slow和fast相遇时就退出
    while(fast && fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;

        // 如果相遇就退出
        if(slow == fast)
            break;
    }

    // 走到这里有可能时slow == fast 也有可能链表不是带环链表
    // 传入空链表的情况也要排除
    // 如果链表只有一个节点且不自身成环的情况要排除
    if(fast == NULL || fast->next == NULL) // 排除不是带环链表的情况
        return NULL;

    // 让相遇点和链表起始点同时开始走,当两个指针相遇的时候,这个节点就是入环的第一个节点
    // 注意!这里是一个推论,切记不可不知道推论怎么来的! 
    // 推论详解  看自己的笔记或者博客。博客地址:
    ListNode* meet = slow;
    while(meet != head)
    {
        meet = meet->next;
        head = head->next;
    }

    // 走到这里meet和head指向的节点就是入环的第一个节点
    return meet;
}

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

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

相关文章

使用protoc-jar-maven-plugin生成grpc项目

在《使用protobuf-maven-plugin生成grpc项目》中我们使用protobuf-maven-plugin完成了grpc代码的翻译。本文我们将只是替换pom.xml中的部分内容&#xff0c;使用protoc-jar-maven-plugin来完成相同的功能。总体来说protoc-jar-maven-plugin方案更加简便。 环境 见《使用proto…

【数据结构(邓俊辉)学习笔记】列表01——从向量到列表

文章目录 0.概述1. 从向量到列表1.1 从静态到动态1.2 从向量到列表1.3 从秩到位置1.4 列表 2. 接口2.1 列表节点2.1.1 ADT接口2.1.2 ListNode模板类 2.2 列表2.2.1 ADT接口2.2.2 List模板类 0.概述 学习了向量&#xff0c;再介绍下列表。先介绍下列表里的概念和语义&#xff0…

global IoT SIM解决方案

有任何关于GSMA\IOT\eSIM\RSP\业务应用场景相关的问题&#xff0c;欢迎W: xiangcunge59 一起讨论, 共同进步 (加的时候请注明: 来自CSDN-iot). Onomondo提供的全球IoT SIM卡解决方案具有以下特点和优势&#xff1a; 1. **单一全球配置文件**&#xff1a;Onomondo的SIM卡拥…

hive-row_number() 和 rank() 和 dense_rank()

row_number() 是无脑排序 rank() 是相同的值排名相同&#xff0c;相同值之后的排名会继续加&#xff0c;是我们正常认知的排名&#xff0c;比如学生成绩。 dense_rank()也是相同的值排名相同&#xff0c;接下来的排名不会加。不会占据排名的坑位。

C#知识|将选中的账号信息展示到控制台(小示例)

哈喽&#xff0c;你好啊&#xff0c;我是雷工&#xff01; 上篇学习了控件事件的统一关联&#xff0c; 本篇通过实例练习继续学习事件统一处理中Tag数据获取、对象的封装及泛型集合List的综合运用。 01 实现功能 在上篇的基础上实现&#xff0c;点击选中喜欢的账号&#xff0…

WebSocket 多屏同显和异显

介绍 多屏同显:通过在一个应用上进行操作之后,另一个应用也能跟着一起发生改变,例如app1播放了晴天这首音乐,那么app2也要同步播放这首音乐,确保所有屏幕显示的内容完全相同。多屏异显:每个屏幕可以显示不同的内容,或者在内容更新时存在一定的延迟,而不需要严格保持同步…

MyBatis 使用 XML 文件映射

在MyBatis中 我们可以使用各种注解来配置我们Mapper 类中的方法 我们为什么要使用XML文件呢&#xff1f; 如果我们是一条非常长的SQL 语句 使用 注解配置的话&#xff0c; 会非常不利于阅读 如下 所以&#xff0c;就需要使用到一个XML文件来对SQL语句进行映射&#xff0c;那么 …

力扣763. 划分字母区间

Problem: 763. 划分字母区间 文章目录 题目描述思路复杂度Code 题目描述 思路 1.创建一个名为 last 的数组&#xff0c;用于存储每个字母在字符串 s 中最后出现的位置。然后&#xff0c;获取字符串 s 的长度 len。 2.计算每个字母的最后位置&#xff1a;遍历字符串 s&#xff0…

(五)SQL系列练习题(上)创建、导入与查询 #CDA学习打卡

目录 一. 创建表 1&#xff09;创建课程表 2&#xff09;创建学生表 3&#xff09;创建教师表 4&#xff09;创建成绩表 二. 导入数据 1&#xff09;导入课程科目数据 2&#xff09;导入课程成绩数据 3&#xff09;导入学生信息数据 4&#xff09;导入教师信息数据 …

RocketMQ SpringBoot 3.0不兼容解决方案

很多小伙伴在项目升级到springBoot3.0版本以上之后&#xff0c;整合很多中间件会有很多问题&#xff0c;下面带小伙伴解决springBoot3.0版本以上对于RocketMQ 不兼容问题 报错信息 *************************** APPLICATION FAILED TO START *************************** Des…

电脑崩溃了,之前备份的GHO文件怎么恢复到新硬盘?

前言 之前咱们说到用WinPE系统给电脑做一个GHO镜像备份&#xff0c;这个备份可以用于硬盘完全崩溃换盘的情况下使用。 那么这个GHO镜像文件怎么用呢&#xff1f; 咱们今天详细来讲讲&#xff01; 如果你的电脑系统硬盘崩溃了或者是坏掉了&#xff0c;那么就需要使用之前备份…

ubuntu 小工具

随着在专业领域待得越久&#xff0c;发现总会遇到各种奇怪的需求。因此&#xff0c;这里介绍一些ubuntu上的一些小工具。 meld 对比文件 sudo apt-get install meld sudo apt --fix-broken install # maybeHow to compare two files

pytorch笔记:ModuleDict

1 介绍 在 PyTorch 中&#xff0c;nn.ModuleDict 是一个方便的容器&#xff0c;用于存储一组子模块&#xff08;即 nn.Module 对象&#xff09;的字典这个容器主要用于动态地管理多个模块&#xff0c;并通过键来访问它们&#xff0c;类似于 Python 的字典 2 特点 组织性 nn…

《金融研究》:普惠金融改革试验区DID工具变量数据(2012-2023年)

数据简介&#xff1a;本数据集包括普惠金融改革试验区和普惠金融服务乡村振兴改革试验区两类。 其中&#xff0c;河南兰考、浙江宁波、福建龙岩和宁德、江西赣州和吉安、陕西铜川五省七地为普惠金融改革试验区。山东临沂、浙江丽水、四川成都三地设立的是普惠金融服务乡村振兴…

可视化大屏应用(20):智慧水务、水利、防汛

hello&#xff0c;我是大千UI工场&#xff0c;本篇分享智智慧水务的大屏设计&#xff0c;关注我们&#xff0c;学习N多UI干货&#xff0c;有设计需求&#xff0c;我们也可以接单。 在智慧水务领域&#xff0c;可视化大屏具有以下几个方面的价值&#xff1a; 数据展示和监控 可…

华为手机 鸿蒙系统-android studio识别调试设备,开启adb调试权限

1.进入设置-关于手机-版本号&#xff0c;连续点击7次 认证&#xff1a;有锁屏密码需要输入密码&#xff0c; 开启开发者配置功能ok 进入开发者配置界面 打开调试功能 重新在androd studio查看可运行running devices显示了&#xff0c; 不行的话&#xff0c;重启一下android …

开源版本管理系统的搭建一:SVN

作者&#xff1a;私语茶馆 1.Windows搭建SVN版本管理系统 1.1.SVN概要和组成 背景介绍 Svn是一个开源版本管理系统&#xff0c;由CollabNet公司于2000年发布&#xff0c;23年12月发布最新版本Apache Subversion 1.14.3。官方网站&#xff1a;Apache Subversion。 Svn可以直…

npm install报错

总结&#xff1a;没有安装visual studio 2017以上带有C桌面开发的版本 #开始试错 #报错总信息mingw_x64_win版本 百度网盘链接: link 提取码&#xff1a;3uou #尝试用mingw配置个C编译器&#xff0c;并配置环境变量&#xff0c;失败 #只认可使用VS!GIthub原址: 链接: link 上…

「 网络安全常用术语解读 」SBOM主流格式CycloneDX详解

CycloneDX是软件供应链的现代标准。CycloneDX物料清单&#xff08;BOM&#xff09;可以表示软件、硬件、服务和其他类型资产的全栈库存。该规范由OWASP基金会发起并领导&#xff0c;由Ecma International标准化&#xff0c;并得到全球信息安全界的支持&#xff0c;如今CycloneD…

thinkphp家政上门预约服务小程序家政保洁师傅上门服务小程序上门服务在线派单安装教程

介绍 thinkphp家政上门预约服务小程序家政保洁师傅上门服务小程序上门服务在线派单安装教程 上门预约服务派单小程序家政小程序同城预约开源代码独立版安装教程 程序完整&#xff0c;经过安装检测&#xff0c;可放心下载安装。 适合本地的一款上门预约服务小程序&#xff0…
最新文章