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

转载请注明出处:http://www.cnblogs.com/wuzetiandaren/p/4251303.html

声明:现大部分文章为寻找问题时在网上相互转载,此博是为自己做个记录记录,方便自己也方便有类似问题的朋友,本文的思想也许有所借鉴,但源码均为本人实现,如有侵权,请发邮件表明文章和原出处地址,我一定在文章中注明。谢谢。

题目:判断一个单链表是否有环,如果有环,求出环的入口节点。

题目分析:

  建一个待头节点的单链表,有两个指针p,q最开始都指向第一个真正节点,p,诶次走1步,q每次走两步,若有环,p和q会在环中的某个节点处相遇。具体分析如下图所示:

    

 

java实现:

  1 package com.interview;
  2
  3 public class _7_1JudgeListCycle {
  4
  5     /**
  6      * @param args
  7      */
  8     public static int x=0, y=0, k=0;   //分别记录圈外链表的长度,圈的长度,第一次相遇时慢指针走的总长度
  9     public static void main(String[] args) {
 10         Node first = createList(24, 6);   //创建有环链表
 11         judge(first);      //判断该单链表是否有环
 12     }
 13
 14     //创建带头节点的链表、参数为:链表总长度,入口节点
 15     public static Node createList(int size, int entry){
 16         /**  1)用尾插法构建单链表,此处构建有环单链表,让node24指向node6*/
 17         Node first = new Node(null,null);    //头节点
 18         Node r = first;   //指向链表的尾节点
 19         Node t = null;    //指向环的入口节点
 20         for(int i=1;i<=size;i++){
 21             Node node = new Node((new Stu(""+i,"小牛犊@"+i)),null);
 22             r.next = node;
 23             r = node;
 24             if(i==entry) t = r;  //让环的入口为节点6
 25         }
 26         r.next = t;       //若是构建无环单链表,此处   r.next = null;
 27         return first;
 28     }
 29
 30     //判断是否有环
 31     public static void judge(Node first){
 32         /**  2)判断是否有环若pq先重合则有环,若q先指向null,则无环    */
 33         Node p = first.next;
 34         Node q = first.next;
 35         String flag = "0";  //标记有环还是无环,0-无环,1-有环
 36         if(p==null){     //空链表
 37             System.out.println("链表为空!");
 38             return;
 39         }
 40         k = 1;
 41         while(q.next!=null && q.next.next!=null){
 42             k++;
 43             p=p.next;
 44             q=(q.next).next;
 45             if(q.equals(p)){
 46                 flag="1";
 47                 System.out.println("p和q相遇了,该单链表有环!");
 48                 break;
 49             }
 50         }
 51
 52         /**  3)有环的时候求环的入口节点    */
 53         if(flag.equals("1")){    //有环
 54             p=first.next; //p重新指向第一个节点
 55             searchEnter(p,q);
 56         }
 57         else{   //是因为q走到了尽头才跳出循环的,无环
 58             System.out.println("该单链表没有环!");
 59         }
 60     }
 61
 62     //求环的入口节点
 63     public static void searchEnter(Node p, Node q){
 64         x=1;
 65         while(!q.equals(p)){
 66             x++;  //求圈外长度
 67             p=p.next;
 68             q=q.next;
 69         }
 70         Node r = q.next;   //为了求圈长设定的
 71         y = 1;     //求圈长度
 72         while(r!=q){
 73             r = r.next;
 74             y++;
 75         }
 76
 77         System.out.println("入口节点是链表中的第"+ x+"个节点!");
 78         System.out.println("入口节点的no: "+ p.stu.no);
 79         System.out.println("入口节点的名字为: "+ p.stu.name);
 80         System.out.println("圈的长度是:"+ y);
 81         System.out.println("第一次相遇时,慢指针在圈内走的长度为:"+ (k-x));
 82         System.out.println("第一次相遇时,快指针在圈内走了:"+ (k/y)+"圈");
 83         System.out.println("第二次相遇时,快指针在圈内又走了:"+ (k/y-1)+"圈");
 84     }
 85
 86     //链表节点的数据结构
 87     static class Node {
 88         public Stu stu;
 89         public Node next;
 90
 91         public Node() {
 92             super();
 93             // TODO Auto-generated constructor stub
 94         }
 95         public Node(Stu stu, Node next) {
 96             super();
 97             this.stu = stu;
 98             this.next = next;
 99         }
100     }
101
102     //节点立里的元素
103     private static class Stu {
104         public String no;
105         public String name;
106         public Stu(String no, String name) {
107             super();
108             this.no = no;
109             this.name = name;
110         }
111     }
112 }

运行效果:

1.无环:

该单链表没有环!

2.有环:

p和q相遇了,该单链表有环!
入口节点是链表中的第6个节点!
入口节点的no: 6
入口节点的名字为: 小牛犊@6
圈的长度是:19
第一次相遇时,慢指针在圈内走的长度为:14
第一次相遇时,快指针在圈内走了:1圈
第二次相遇时,快指针在圈内又走了:0圈

时间: 2024-10-13 05:46:04

7_1判断一个单链表是否有环的相关文章

判断一个单链表是否有环

用快慢指针法,慢指针一次走一步,快指针一次走两步,如果有环必定相遇 这里给出C++和Python 版 Python: """ Definition of ListNode class ListNode(object): def __init__(self, val, next=None): self.val = val self.next = next """ class Solution: """ @param h

单链表是否有环问题

小咖是某计算机专业的学生.linglingling,伴随这铃声,同学们都准时来上课了. 老师:这节课是计算机课,我们学习了单链表,请同学说一下什么是单链表. 同学们纷纷举起了手.好,小白你来说. 小白:单链表是一种线性表,它的存储单元可以不是连续的,从头节点开始,每个节点都有一个指针,指向下一个节点,最后一个节点指向null,头节点不被指向. 老师:好,那么谁来说一下链表的这种结构有什么优点呢? 同学们纷纷举起了手.好,小黄你来说. 小黄:线性表的一种,链式表的在存储空间中可以是不连续的,因此存

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

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

判断单链表是否有环以及环的连接点

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

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

原文: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,

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终点