检查链表是否有环,返回值为bool和从头节点进入环的第一个节点两种情况

本文主要是介绍检查链表是否有环,返回值为bool和从头节点进入环的第一个节点两种情况,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目1(不返回节点)

给定单链表,检查链表是否有环。

代码实现:

bool IsCircle(List plist)
{assert(plist != NULL);if (plist == NULL||plist->next==NULL)return false;Node* p = plist->next;//慢指针,一次走一步Node* q = plist->next->next;//快指针,一次走两步while (q!= NULL&&q->next!=NULL){if (p == q){break;}else{p = p->next;q = q->next->next;}}if (q == NULL || q->next == NULL){return false;}else{return true;}}

题目2(返回节点)

给定单链表(head),如果有环的话请返回从头节点进入环的第一个节点。

思路: 

给定慢指针p->next;快指针q->next->next。如图,假设快、慢指针在点3相遇,快指针走过的路径是慢指针的两倍。

慢指针走过的路径长度:m+n
快指针走过的路径长度:m+c*x+n
快指针走过的路径是慢指针的2倍,所以,2*(m+n)=m+c*x+n

由上式得到m=c*x-n=c*x-n+c-c=c*(x-1)+c-n,即m的长度比圆的周长倍数多出c-n的长度,相遇点加上c-n的位置就是进入环的第一个节点。

代码实现:

Node* FirstCircleNode(List plist)
{assert(plist != NULL);if (plist == NULL||plist->next==NULL)return NULL;Node* p = plist->next;//慢指针Node* q = plist->next->next;//快指针for (p, q; q != NULL && q->next != NULL; p = p->next, q = q->next->next){if (p == q)break;}if (q==NULL||q->next==NULL)//没有环{return NULL;}//快慢指针相遇了Node* s = plist;//1while (s != q){s = s->next;q = q->next;}return s;
}int main()
{Node head;InitList(&head);for (int i = 0; i < 10; i++){Insert_tail(&head, i);}Show(&head);Node* p1 = FirstCircleNode(&head);if (p1!=NULL)printf("有环,%d\n",p1->data);elseprintf("无环\n");//Node* q = Search(&head, 3);//链表中的某一个节点(数据域为3)//Node* s = Search(&head, 9);//尾巴节点//s->next = q;//Node* p2 = IsCircle(&head);//if (p2 != NULL)//	printf("有环,%d\n", p2->data);//else//	printf("无环\n");Node* q = Search(&head, 5);//链表中的某一个节点(数据域为5)Node* s = Search(&head, 9);//尾巴节点s->next = q;Node* p2 = FirstCircleNode(&head);if (p2 != NULL)printf("有环,%d\n", p2->data);elseprintf("无环\n");return 0;
}

 

这篇关于检查链表是否有环,返回值为bool和从头节点进入环的第一个节点两种情况的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/771768

相关文章

Go使用pprof进行CPU,内存和阻塞情况分析

《Go使用pprof进行CPU,内存和阻塞情况分析》Go语言提供了强大的pprof工具,用于分析CPU、内存、Goroutine阻塞等性能问题,帮助开发者优化程序,提高运行效率,下面我们就来深入了解下... 目录1. pprof 介绍2. 快速上手:启用 pprof3. CPU Profiling:分析 C

C++实现回文串判断的两种高效方法

《C++实现回文串判断的两种高效方法》文章介绍了两种判断回文串的方法:解法一通过创建新字符串来处理,解法二在原字符串上直接筛选判断,两种方法都使用了双指针法,文中通过代码示例讲解的非常详细,需要的朋友... 目录一、问题描述示例二、解法一:将字母数字连接到新的 string思路代码实现代码解释复杂度分析三、

MySQL进阶之路索引失效的11种情况详析

《MySQL进阶之路索引失效的11种情况详析》:本文主要介绍MySQL查询优化中的11种常见情况,包括索引的使用和优化策略,通过这些策略,开发者可以显著提升查询性能,需要的朋友可以参考下... 目录前言图示1. 使用不等式操作符(!=, <, >)2. 使用 OR 连接多个条件3. 对索引字段进行计算操作4

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

C#比较两个List集合内容是否相同的几种方法

《C#比较两个List集合内容是否相同的几种方法》本文详细介绍了在C#中比较两个List集合内容是否相同的方法,包括非自定义类和自定义类的元素比较,对于非自定义类,可以使用SequenceEqual、... 目录 一、非自定义类的元素比较1. 使用 SequenceEqual 方法(顺序和内容都相等)2.

查询Oracle数据库表是否被锁的实现方式

《查询Oracle数据库表是否被锁的实现方式》本文介绍了查询Oracle数据库表是否被锁的方法,包括查询锁表的会话、人员信息,根据object_id查询表名,以及根据会话ID查询和停止本地进程,同时,... 目录查询oracle数据库表是否被锁1、查询锁表的会话、人员等信息2、根据 object_id查询被

shell脚本快速检查192.168.1网段ip是否在用的方法

《shell脚本快速检查192.168.1网段ip是否在用的方法》该Shell脚本通过并发ping命令检查192.168.1网段中哪些IP地址正在使用,脚本定义了网络段、超时时间和并行扫描数量,并使用... 目录脚本:检查 192.168.1 网段 IP 是否在用脚本说明使用方法示例输出优化建议总结检查 1