链表中环的入口节点

题目描述

给一个链表,若其中包含环,请找出该链表的环的入口结点,不然,输出null。java

/*
 public class ListNode {
    int val;
    ListNode next = null;

    ListNode(int val) {
        this.val = val;
    }
}
*/
public class Solution {

    public ListNode EntryNodeOfLoop(ListNode pHead){
        
    }
}

解法一

遍历链表的时候,用一个容器list依次装入链表的节点,若是发现有重复的节点,那么就是链表的环的入口节点oop

import java.util.ArrayList;
public class Solution {

    public ListNode EntryNodeOfLoop(ListNode pHead){
        ArrayList<ListNode> list = new ArrayList<>();
        while (!list.contains(pHead) && pHead != null){
            list.add(pHead);
            pHead = pHead.next;
        }

        return pHead;
    }
}

时间复杂度:O(n^2)this

解法二

采用双指针的方法(这个方法还能够用来判断链表中有没有环),一个快指针一个慢指针,快指针一次走两步,慢指针一次走一步,若是链表中有环,那么两个指针必定就能够相遇,而且相遇的地方,必定在环的某处spa

设相遇的地方距环的入口点一边有a个节点,一边有b个节点,链表除环之外有x个节点,环中一共有c个节点(c=a+b).net

那么能够获得以下关系式,用b = c -a表示指针

![图片描述][1]code

public ListNode EntryNodeOfLoop(ListNode pHead){
        if(pHead == null || pHead.next == null)return  null;
        ListNode slow = pHead.next;
        ListNode fast = pHead.next.next;
        while (slow != fast && pHead != null){
            slow = slow.next;
            fast = fast.next.next;
        }
        slow = pHead;
        while(slow != fast){
            slow = slow.next;
            fast = fast.next;
        }
        return slow;
    }

该方法的时间复杂度为O(n)blog

参考

《剑指offer》——链表中环的入口结点图片

相关文章
相关标签/搜索