随笔 - 301  文章 - 2  评论 - 16  阅读 - 34万

剑指Offer——链表的环以及环的入口

在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针。 例如,链表1->2->3->3->4->4->5 处理后为 1->2->5

解题思路

这个题包含两个子问题,一个是判断链表是否有环,另一个是如果有环,找到环的入口。

  • 对于如何判断链表有环问题,较为基础:快慢指针。

  • 对于如何找到环的入口,这个地方需要推导一下:

    x表示头结点到入口结点的距离,表示入口结点到fast和slow相遇结点的距离,z是环中剩下的长度,即有z+y=L。L表示的是环的长度。那么当快慢指针相遇时候有:S = x+y 2*S = x+y+nL。即,慢指针走过x+y的结点,快指针比慢指针多走了n圈。这里慢指针在环中也可能多走了几圈才相遇的,但是可以不用表示出来。因为快指针的nL表示的就是比慢指针多走的圈数。举例:慢指针如果走 x+y+2L,快指针走 x+y+4L 那么就可以写成:慢x+y 快x+y+2L.是等价的。
    \begin{cases} 2s = x+y+nL \\ s = x+y \end{cases}
    消去s,可得x+y = nL,同时:z+y = L,代入消去y,可得x=(n-1)L+z

    如图

    得到x = (n-1)L+z 能说明什么问题呢?

    走x步,等于走 n-1圈再走上z步。走x步,正好就是从头结点到环入口结点。而z步是从相遇地方,到入口结点,在加上多少圈,实际走的都是从相遇到入口结点。举个例子,如果n = 1,那么x=z。如果n=2,那么x=L+z.就是多走一圈到相遇结点,然后再走z步还是到环的入口结点。

    这个时候,我们只需要令一个结点从头结点出发,走x步,另一个结点从相遇结点出发,走n圈(n=0,1,2,...)再走z步,他们会同时到达环的入口地方。

    题解

    复制代码
    public ListNode EntryNodeOfLoop(ListNode pHead)
        {
            if(pHead == null) return null;
            if (isLoop(pHead)){
                ListNode slow = pHead.next;
                ListNode fast = pHead.next.next;
    
                while(slow != fast){
                    slow = slow.next;
                    fast = fast.next.next;
                }
                // 循环结束时候  slow == fast
                slow = pHead;
                // 一个从头开始走 ,一个从相遇地方开始走,走同样的步数
                // 因为 x= z+(n-1)L
                while(slow != fast){
                    slow = slow.next;
                    fast = fast.next;
                }
                return slow;
            }
            else  return null;
        }
        public boolean isLoop(ListNode pHead){
            if (pHead.next == null) return false;
            ListNode slow = pHead;
            ListNode fast = pHead;
            while (fast != null && fast.next !=null){
                slow = slow.next;
                fast = fast.next.next;
                if (slow == fast){
                    return true;
                }
            }
            return false;
        }
    复制代码

     

    posted on   Code2020  阅读(108)  评论(0编辑  收藏  举报
    编辑推荐:
    · .NET Core 中如何实现缓存的预热?
    · 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
    · AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
    · 基于Microsoft.Extensions.AI核心库实现RAG应用
    · Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
    阅读排行:
    · TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
    · 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
    · 【译】Visual Studio 中新的强大生产力特性
    · 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
    · 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
    < 2025年3月 >
    23 24 25 26 27 28 1
    2 3 4 5 6 7 8
    9 10 11 12 13 14 15
    16 17 18 19 20 21 22
    23 24 25 26 27 28 29
    30 31 1 2 3 4 5

    点击右上角即可分享
    微信分享提示