Floyd 判圈算法(Floyd Cycle Detection Algorithm)

news2025/1/9 15:24:11

Floyd 判圈算法(Floyd Cycle Detection Algorithm)

  1. 前言

Floyd判圈算法属于对指针操作的算法,它一般需要且仅需要两个指针,通过设定不同的指针移动速度,来判定链表或有限状态机中是否存在环。人为规定移动较快的指针称为快速指针(fast pointer),移动较慢的指针称作慢速指针(slow pointer),快速指针比慢速指针移动速度快2倍。

  1. 链表中环判定

在判定链表中是否存在环过程中,快慢指针可能会出现下列两种情形之一:

a) 快速指针指向到达链表的结尾 (fast pointer =NULL),此种情况表明链表中不存在环。这里需要说明的是,快速指针一定先于慢速指针道道链表的尾部。

b) 快速指针在遍历过程中,赶上慢速指针,并且与慢速指针相会和。这种情况下表明链表中存在着环(Loop)。

具体看一个例子,显而易见,链表中存在一个环,环的起点为5,应用快慢指针对此链表进行遍历,并判断其是否存在环。

在这里插入图片描述

第一步,建立快慢指针两个变量,并且令快慢指针均指向头节点,准备开始以不同的速度进行移动指针,观察会出现上述提到的哪类情形。

在这里插入图片描述

令快指针以两倍于慢指针的速度向前移动

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

显而易见,当移动至第七个元素的时候,快指针赶上慢指针,在此处快慢指针相遇。符合第二类情形,所以判定链表中存在着环。

  1. 算法逻辑

如果存在环,那么快慢指针一定在某处相遇,那究竟是为什么呢? 借助相关图示进行数学分析和理解,

在这里插入图片描述

上图中,采用链表两个节点之间的数目来代表距离

X-代表头节点(Head)到 环起点 之间的距离(Loop start point)

Y-代表环起点(Loop start point)和两指针的第一次相遇点(Meet point)之间的距离

C-代表环的周长/距离

当快慢指针在相遇的时候,分别求出快慢指针移动的相应距离,慢指针移动的距离:
D i s t a n c e _ s l o w = X + Y + s ∗ C ; s ∈ n o n − n e g a t i v e   i n t e g e r ( 非负整数 ) Distance\_slow=X+Y+s*C; s∈ non-negative\ integer(非负整数) Distance_slow=X+Y+sC;snonnegative integer(非负整数)

与之对应的是快指针移动的距离:
D i s t a n c e _ f a s t = X + Y + f ∗ C ; f ∈ n o n − n e g a t i v e   i n t e g e r ( 非负整数 ) Distance\_fast=X+Y+f*C; f∈ non-negative\ integer(非负整数) Distance_fast=X+Y+fC;fnonnegative integer(非负整数)
在相同移动次数的前提下,快指针移动的距离为慢指针移动距离的两倍,利用此条件可以对上面的距离建立有效的关系式。
D i s t a n c e _ f a s t = 2 ∗ D i s t a n c e _ s l o w Distance\_fast=2*Distance\_slow Distance_fast=2Distance_slow
通过等式变换和合并同类项后,我们可以得到,
X + Y + f ∗ C = 2 ∗ ( X + Y + s ∗ C ) X+Y+f*C=2*(X+Y+s*C) X+Y+fC=2(X+Y+sC)

X + Y = f ∗ C − 2 ∗ s ∗ C = ( f − 2 ∗ s ) ∗ C = K ∗ C X+Y=f*C-2*s*C=(f-2*s)*C=K*C X+Y=fC2sC=(f2s)C=KC

通过简化上面的关系是,最终我们可以得到两个简单的关系式:

X+Y=K*C----(1)

X=K*C-Y----(2)

上面关系式为寻找环的起点提供了方向和思路,如果两个指针相遇后,把头节点赋给慢指针,快指针的位置保持不变,我们以每次步长为1进行移动,快慢指针最后将相遇在环的起点位置(Loop start point)。此时慢指针走过的距离为X,快指针走过的距离为K*C-Y=X。

  1. 算法实现

a) 头文件声明函数, make_node 函数从整数创建待插入的结点;insert_node 利用头插法,插入结点,头结点一直在第一个位置。is_contain_loop函数判定链表中是否存在环,find_loop_start_point函数找到环的起始结点。

/**
 * @file floyd_cycle_detecion.h
 * @author your name (you@domain.com)
 * @brief 
 * @version 0.1
 * @date 2023-06-18
 * 
 * @copyright Copyright (c) 2023
 * 
 */
#ifndef FLOYD_CYCLE_DETECTION_H
#define FLOYD_CYCLE_DETECTION_H
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdbool.h>

typedef struct node
{
    int data;
    struct node *next;
}node,*node_link;


void make_node(node_link *new_node, int new_value);

void insert_node(node_link *head,int new_value);

bool is_contain_loop(node_link *head);

node_link find_loop_start_point(node_link *head);


#endif

b) 函数定义

值得一提的是,快指针采用 fast_pointer=fast_pointer->next->next方式实现跨越式移动。

/**
 * @file floyd_cycle_detecion.c
 * @author your name (you@domain.com)
 * @brief 
 * @version 0.1
 * @date 2023-06-18
 * 
 * @copyright Copyright (c) 2023
 * 
 */

#ifndef FLOYD_CYCLE_DETECTION_C
#define FLOYD_CYCLE_DETECTION_C
#include "floyd_cycle_detection.h"

void make_node(node_link *new_node, int new_value)
{
    *new_node=(node_link)malloc(sizeof(node));
    (*new_node)->data=new_value;
    (*new_node)->next=NULL;

    return;
}

void insert_node(node_link *head, int new_value)
{
    node_link new_node;

    make_node(&new_node,new_value);

    if((*head)==NULL)
    {
        *head=new_node;
    }
    else //insert before the current node
    {
        new_node->next=*head;
        *head=new_node;
    }

    return;
}

bool is_contain_loop(node_link *head)
{
    node_link slow_pointer;
    node_link fast_pointer;

    slow_pointer=fast_pointer=*head;

    while(slow_pointer && fast_pointer && fast_pointer->next)
    {
        slow_pointer=slow_pointer->next;
        fast_pointer=fast_pointer->next->next;

        if(slow_pointer==fast_pointer)
        {
            return true;
        }
    }

    return false;
}

node_link find_loop_start_point(node_link *head)
{
    node_link slow_pointer;
    node_link fast_pointer;

    slow_pointer = fast_pointer = *head;

    while (slow_pointer && fast_pointer && fast_pointer->next)
    {
        slow_pointer = slow_pointer->next;
        fast_pointer = fast_pointer->next->next;

        if (slow_pointer == fast_pointer)
        {
            break;
        }
    }

    if(slow_pointer!=fast_pointer)
    {
        return NULL;
    }

    slow_pointer=*head;

    while(slow_pointer!=fast_pointer)
    {
        slow_pointer=slow_pointer->next;
        fast_pointer=fast_pointer->next;
    }

    return slow_pointer;
}

#endif

c) 测试函数

通过临时结点temp_node_1和temp_node_2在链表内创建一个环,方便后面的相关测试。

/**
 * @file floyd_cycle_detetcion_main.c
 * @author your name (you@domain.com)
 * @brief 
 * @version 0.1
 * @date 2023-06-18
 * 
 * @copyright Copyright (c) 2023
 * 
 */
#ifndef FLOYD_CYCLE_DETECTION_MAIN_C
#define FLOYD_CYCLE_DETECTION_MAIN_C
#include "floyd_cycle_detection.c"

int main(void)
{
    int i;
    int arr[]={10,9,8,7,6,5,4,3,2,1};
    int n=sizeof(arr)/sizeof(int);

    node_link head=NULL;
    node_link temp_node_1;
    node_link temp_node_2;
    node_link loop_start_point;
    bool res;

    for(i=0;i<n;i++)
    {
        insert_node(&head,arr[i]);
    }

    temp_node_1=head;

    while(temp_node_1 && temp_node_1->data!=5)
    {
        temp_node_1=temp_node_1->next;
    }

    temp_node_2=temp_node_1;

    while(temp_node_2->next!=NULL)
    {
        temp_node_2=temp_node_2->next;
    }

    temp_node_2->next=temp_node_1;

    res=is_contain_loop(&head);

    if(res)
    {
        printf("there is a loop inside\n");
    }
    else
    {
        printf("there is no a loop inside\n");
    }

    loop_start_point=find_loop_start_point(&head);

    printf("The loop start point had been detected\n");
    getchar();

    return EXIT_SUCCESS;
}



#endif

  1. 小结

本文通过学习Floyd判圈算法,对快慢指针有了更加深入的理解,希望今后能更加数量运用快慢指针的思想,分析解决不同的实际问题。

参考资料:

tf(“there is no a loop inside\n”);
}

loop_start_point=find_loop_start_point(&head);

printf("The loop start point had been detected\n");
getchar();

return EXIT_SUCCESS;

}

#endif


5. 小结

本文通过学习Floyd判圈算法,对快慢指针有了更加深入的理解,希望今后能更加数量运用快慢指针的思想,分析解决不同的实际问题。



参考资料:

1.[Floyd’s Cycle Finding Algorithm - GeeksforGeeks](https://www.geeksforgeeks.org/floyds-cycle-finding-algorithm/)

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

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

相关文章

Java官方笔记9Lambda表达式

Lambda Expression 有了Lambda Expression&#xff0c;就不用再写anonymous classes。 写Lambda&#xff0c;首先要找到它的类型。 There is a restriction on the type of a lambda expression: it has to be a functional interface. 函数接口&#xff0c;只有1个抽象方法的接…

Vue中v-text、v-html、v-on的基本语法(二)

文章目录 前言一、vue中data属性定义对象、数组相关数据二、v-text、v-html指令使用三、v-on基本指令使用(一)四、v-on指令基本使用(二)之在函数中获取vue实例本身this五、v-on指令基本使用(二)之在函数中传递参数六、v-on指令基本使用(二)之简化写法绑定函数和事件定义的两种写…

从零搭建一台基于ROS的自动驾驶车-----2.运动控制

系列文章目录 北科天绘 16线3维激光雷达开发教程 基于Rplidar二维雷达使用Hector_SLAM算法在ROS中建图 Nvidia Jetson Nano学习笔记–串口通信 Nvidia Jetson Nano学习笔记–使用C语言实现GPIO 输入输出 Autolabor ROS机器人教程 从零搭建一台基于ROS的自动驾驶车-----1.整体介…

Unreal 5 实现丧尸伤害和死亡

这一篇主要是实现玩家攻击丧尸可以造成伤害和自身血量为零时&#xff0c;丧尸可以死亡。丧尸也可以对玩家造成伤害&#xff0c;有攻击范围的判定。 这一篇的功能实现有四个功能&#xff1a; 丧尸被攻击掉血丧尸死亡处理玩家被攻击掉血玩家死亡处理 丧尸被攻击掉血 子弹的修改…

C语言内存操作函数,memcpy的使用和模拟实现,memmove的使用和模拟实现,memcmp的使用,memset的使用。

1.memcpy 函数原型&#xff1a; void *memcpy( void *dest, const void *src, size_t count );void *dest 目标数据首元素地址const void *src 源数据(需要拷贝过去的数据)size_t count 需要拷贝数据的字节大小void *memcpy 拷贝结束后&#xff0c;返回目标数据的起始地址 函…

【简单的图像信息展示应用程序】PYQt5

写在前面的话 这段代码的作用是创建一个简单的图像信息展示应用程序&#xff0c;用户可以点击按钮查看特定文件夹中图像的文件名、大小&#xff0c;并通过查看按钮查看图像。请注意&#xff0c;文件夹路径需要根据实际情况进行修改。 代码讲解 这段代码是使用PyQt5库创建一个…

特征选择:过滤法,嵌入法,包装法

特征选择时首先要去除冗余特征。 它是由其他其他的特征中推演出来的。比如&#xff0c;一个球的体积&#xff0c;那么半径这个特征就是冗余的&#xff0c;因为我们可以由球的体积推算半径。冗余特征在很多时候都是不起作用的 过滤法 过滤方法通常用作预处理步骤&#xff0c;特…

c++11 标准模板(STL)(std::basic_ios)(三)

定义于头文件 <ios> template< class CharT, class Traits std::char_traits<CharT> > class basic_ios : public std::ios_base 类 std::basic_ios 提供设施&#xff0c;以对拥有 std::basic_streambuf 接口的对象赋予接口。数个 std::basic_ios…

2013年全国硕士研究生入学统一考试管理类专业学位联考英语(二)试题

2013考研英语&#xff08;二&#xff09;真题 Section I Use of English Directions: Read the following text. Choose the best word(s) for each numbered blank and mark A, B, C or D on ANSWER SHEET 1. (10 points) Given the advantages of electronic money, you mi…

某农业大学数据结构A-第2周作业

1.两个顺序表集合的差集 【问题描述】两个顺序表集合的差集 【样例输入】 25 33 57 60 48 9 13 0 12 50 23 60 4 34 25 13 0 【样例输出】 33 57 48 9 【注意】0代表输入的结束&#xff1b;可以用C风格实现&#xff0c;也可以用C风格实现&#xff0c;两种风格大家均需掌握 #…

Kafka架构

5.kafka系统的架构 5.1主题topic和分区partition topic Kafka中存储数据的逻辑分类&#xff1b;你可以理解为数据库中“表”的概念&#xff1b; 比如&#xff0c;将app端日志、微信小程序端日志、业务库订单表数据分别放入不同的topic partition分区&#xff08;提升kafka吞…

【Proteus仿真】常用器件名称

前言 我常用的仿真器件加上收集的&#xff0c;基于Proteus8.13版本。以下分为两部分&#xff0c;内容都一样&#xff0c;一部分是纯文字&#xff0c;一部分是文字图片&#xff0c;方便快速获取和定位。等积累了更多的器件后会在更新的。搜索时可以用CtrlF快速查找。 命名的规则…

管理类联考——英语二——技巧篇——写作——B节——议论文——必备替换句型

议论文必备替换句型 (一&#xff09;表示很明显/众所周知的句型 It is obvious thatIt is clear thatIt is apparent thatIt is evident thatlt is self-evident thatIt is manifest thatIt is well-knownIt is known to all thatIt is widely-accepted thatIt is crystal-cl…

三层交换机互联互通配置 华为交换机

#三层交换机互联互通 交换机配置 命令 #进入系统视图 <Huawei>system-view #关闭系统提示信息 [Huawei]undo info-center enable #创建三个Vlan10 [Huawei]vlan 10 [Huawei-vlan10]quit [Huawei]vlan 20 [Huawei-vlan20]quit [Huawei]vlan 30 [Huawei-vlan30]quit #接…

2023六月第二周(juc知识点记录)

1、多线程资源竞争 先创建资源类&#xff0c;并给线程加锁&#xff0c;推荐使用reatrantlock&#xff0c;然后写业务&#xff0c; 加锁了说明肯定要有执行条件&#xff0c;例如抢到锁还要判断库存必须大于0&#xff0c; 否则线程等待await,然后执行业务&#xff0c;最后创建多个…

一分钟图情论文:《运用服务场景模型理解挪威学术图书馆中的学生体验》

一分钟图情论文&#xff1a;《Applying the servicescape model to understand student experiences of a Norwegian academic library》 1981年&#xff0c;Booms, B. H.和Bitner, M. J.1在一篇市场营销学科的论文中引入了Servicescape模型&#xff0c;用于描述和分析服务场所…

无线供电原理、种类及应用方案介绍

目录 一、无线供电种类 二、无线供电的原理 磁共振技术的电路组成 微波能量传输 三、无线供电的优势 四、市场应用 五、市场趋势 参考文献 一、无线供电种类 包含电磁场耦合能量传输&#xff08;电磁磁感应&#xff08;近场供电&#xff09;、磁共振&#xff08;远场供…

easyX库颜色模型和颜色及样式设置相关函数(注释版)

0.颜色模型和颜色及样式设置相关函数概览 本次我给你带了easyX库系列的颜色模型和样式设置的相关函数&#xff0c;希望您能看得开心。 函数或数据类型描述LINESTYLE画线样式对象。FILLSTYLE填充样式对象。setbkcolor设置当前设备绘图背景色。setlinecolor设置当前设备画线颜色…

[易语言][原创]使用易语言部署yolov8的onnx模型

易语言部署yolo系列模型&#xff0c;现在网上有很多但是他们不够简洁也不够专业&#xff0c;有人专门把opencv封装成易语言支持库然后用opencv在易语言端写&#xff0c;其实这种效率没有在C直接推理效率高&#xff0c;因为易语言往C传递图像数据集是需要转换图像数据集格式才能…

十一、docker学习-docker核心之docker网络(1)

docker网络 当开始大规模使用docker时&#xff0c;你会发现需要了解很多关于网络的知识。docker作为目前最火的轻量级容器技术&#xff0c;有很多令人称道的功能&#xff0c;如docker的镜像管理。然而&#xff0c;docker同样有着很多不完善的地方&#xff0c;网络方面就是Dock…