单链表是否有环问题

小咖是某计算机专业的学生。linglingling,伴随这铃声,同学们都准时来上课了。

老师:这节课是计算机课,我们学习了单链表,请同学说一下什么是单链表。

同学们纷纷举起了手。好,小白你来说。

小白:单链表是一种线性表,它的存储单元可以不是连续的,从头节点开始,每个节点都有一个指针,指向下一个节点,最后一个节点指向null,头节点不被指向。

老师:好,那么谁来说一下链表的这种结构有什么优点呢?

同学们纷纷举起了手。好,小黄你来说。

小黄:线性表的一种,链式表的在存储空间中可以是不连续的,因此存储密度较低,还有增删的时候只需要改变指针的指向就可以了,而不是像顺序表那样,比如你要在某个位置添加一个元素,那么其他的元素都要向后移,因此链式表增删相对较快些。

老师:好,大家掌握的不错啊,那么大家思考一个问题,怎么判断一个单链表是否有环。

如图,这样的单链表就是有环的链表。

注:数字1-10代表节点1-10,节点存储值和指针。

老师:好,小红同学你来说。

小红:两层循环,第一层循环从头遍历,第二层循环从头遍历到第一层循环当前的节点,如果存在相同的节点,那么链表有环。

老师:好,这样做确实是可以判断是否有环,但是效率并不高,遍历两次时间复杂度o(n^2),由于没有开辟额外的空间,所以空间复杂度o(1)。哪位同学还有更好的方法吗。

老师:好,小明你来说一下。

小明:一层循环,创建一个hashset,把节点放到hashset中,如果出现相同的节点,说明单链表有环。

老师:好,小明同学采用了空间换时间的方法,此时的时间复杂度o(n),由于开辟了和链表单独相关的n的空间,所以空间复杂度o(n)。那还有没有更好的方法呢。

老师:谁是计算机课带表啊。

课带表:

老师:那么你来说说,有什么更好的方法。

课代表:

如果两个人在环形的跑道上赛跑,快的人总会再次追上慢的。

因此我们可以假设用两个指针比如两个赛跑的同学,一个快指针,一个慢指针,我们称为快慢指针方法。

那么我们来说一下说它是实现思路:

快指针每次移动两步,慢指针每次移动一步,如果两个节点相等,则说明有环,返回true,否则返回false。

代码实现大概是是这个样子的。

Node  fast = head;

Node  slow = head;

while(fast.next != null)

{

fast = fast.next.next;

slow = slow.next;

if (slow == fast)

return true;

}

return false;

老师:非常好,这个方法的时间复杂度为o(n),空间复杂度为o(1)。请大家回去思考一个问题,如何判断单链表是否有交点。下节课老师提问,同学们下课。

欢迎大家和作者一起交流技术,或者您有好的建议等等,都可以联系作者。

感兴趣的朋友可以关注公众号,更多精彩等着您:javaxiaokashow

原文地址:https://www.cnblogs.com/playburst/p/9097132.html

时间: 2024-10-22 22:53:59

单链表是否有环问题的相关文章

如何判断单链表是否存在环

原文:http://blog.csdn.net/liuxialong/article/details/6555850 如何判断单链表是否存在环 给定一个单链表,只给出头指针h: 1.如何判断是否存在环? 2.如何知道环的长度? 3.如何找出环的连接点在哪里? 4.带环链表的长度是多少? 解法: 1.对于问题1,使用追赶的方法,设定两个指针slow.fast,从头指针开始,每次分别前进1步.2步.如存在环,则两者相遇:如不存在环,fast遇到NULL退出. 2.对于问题2,记录下问题1的碰撞点p,

7_1判断一个单链表是否有环

转载请注明出处:http://www.cnblogs.com/wuzetiandaren/p/4251303.html 声明:现大部分文章为寻找问题时在网上相互转载,此博是为自己做个记录记录,方便自己也方便有类似问题的朋友,本文的思想也许有所借鉴,但源码均为本人实现,如有侵权,请发邮件表明文章和原出处地址,我一定在文章中注明.谢谢. 题目:判断一个单链表是否有环,如果有环,求出环的入口节点. 题目分析: 建一个待头节点的单链表,有两个指针p,q最开始都指向第一个真正节点,p,诶次走1步,q每次走

单链表是否有环的问题解决与讨论(java实现)

单链表是否有环的问题经常在面试中遇到,一般面试中会要求空间为O(1);再者求若有环,则求环产生时的起始位置. 下面采用java实现. //单链表class ListNode{ int val; ListNode next; ListNode(int x){ val=x; next=null; } } public class SearchCycleNode{ ListNode equalListNode=null;//来记录判断有环时出现的相等的时的节点,姑且叫"相遇"节点. //从&

如何判断单链表是否存在环 & 判断两链表是否相交

给定一个单链表,只给出头指针h: 1.如何判断是否存在环? 2.如何知道环的长度? 3.如何找出环的连接点在哪里? 4.带环链表的长度是多少? 解法: 1.对于问题1,使用追赶的方法,设定两个指针slow.fast,从头指针开始,每次分别前进1步.2步.如存在环,则两者相遇:如不存在环,fast遇到NULL退出. 2.对于问题2,记录下问题1的碰撞点p,slow.fast从该点开始,再次碰撞所走过的操作数就是环的长度s. 3.问题3:有定理:碰撞点p到连接点的距离=头指针到连接点的距离,因此,分

求有环单链表中的环长、环起点、链表长

1.判断单链表是否有环 使用两个slow, fast指针从头开始扫描链表.指针slow 每次走1步,指针fast每次走2步.如果存在环,则指针slow.fast会相遇:如果不存在环,指针fast遇到NULL退出. 就是所谓的追击相遇问题: 2.求有环单链表的环长 在环上相遇后,记录第一次相遇点为Pos,之后指针slow继续每次走1步,fast每次走2步.在下次相遇的时候fast比slow正好又多走了一圈,也就是多走的距离等于环长. 设从第一次相遇到第二次相遇,设slow走了len步,则fast走

[CareerCup] 2.6 Linked List Cycle 单链表中的环

2.6 Given a circular linked list, implement an algorithm which returns the node at the beginning of the loop.DEFINITIONCircular linked list: A (corrupt) linked list in which a node's next pointer points to an earlier node, so as to make a loop in the

11.判断单链表是否有环

判断单链表是否有环:   这里也是用到两个指针,如果一个链表有环,那么用一个指针去遍历,是永远走不到头的.   因此,我们用两个指针去遍历:first指针每次走一步,second指针每次走两步,如果first指针和second指针相遇,说明有环.时间复杂度为O (n). 方法 // 方法:检测单链表是否有环 public boolean hasCycle(Node head) { if (head == null) { return false; } Node first = head; Nod

判断单链表是否有环及寻找环的入口

一.判断单链表是否有环 1.遍历链表 遍历链表,将已经访问过的结点,设置为已访问,如果访问同一结点两次,则说明有环,时间O(n),空间O(n). 2.链表反转 遍历链表,将访问的结点指针反向.如果存在环,反转后next指向原链表头,但链表反转会破坏链表的结构,反转需要注意缓存当前结点的下一结点,时间O(n),空间(1). 3.快慢指针 设置快慢两个指针fast和slow,两个指针都链表头开始,fast每次移动2步,slow为1步.如果存在环,则fast先进入环,slow后进入环,最后两者在环中相

判断单链表是否有环相关问题(转载加总结)

给定一个单链表,只给出头指针h: 1.如何判断是否存在环? 2.如何知道环的长度? 3.如何找出环的连接点在哪里? 4.带环链表的长度是多少? 解法: 1.对于问题1,使用追赶的方法,设定两个指针slow.fast,从头指针开始,每次分别前进1步.2步.如存在环,则两者相遇:如不存在环,fast遇到NULL退出. 2.对于问题2,记录下问题1的碰撞点p,slow.fast从该点开始,再次碰撞所走过的操作数就是环的长度s. 3.问题3:有定理:碰撞点p到连接点的距离=头指针到连接点的距离,因此,分

8.判断单链表是否有环(6形状)?如何找到环的“起始”点?如何知道环的长度?

8.判断单链表是否有环(6形状)?如何找到环的“起始”点?如何知道环的长度? 思路: 注意分析题意,题意并非是说单链表完全成O形状的环,而是说单链表成6形状的环. 首先判断是否有环:为此我们建立两个指针,从Head一起向前跑,一个步长为1,一个步长为2,用 while(直到步长2的跑到结尾){检查两个指针是否相等,直到找到为止.} 来进行判断. 原因是,在这场跑步中,结束循环有两种可能,其一是原来无环,所以2先跑到结尾,因为2比1快,二者不可能相等.其二是原来是有环的,因为这场赛跑永远没有z终点