力扣 快慢指针

1 环形链表

141. 环形链表 - 力扣(LeetCode)

定义两个指针,一快一慢。慢指针每次只移动一步,而快指针每次移动两步。初始时,慢指针和快指针都在位置 head,这样一来,如果在移动的过程中,快指针反过来追上慢指针,就说明该链表为环形链表。

使用 do-while 循环,可以把快慢指针的初始值都置为 head

cpp 复制代码
class Solution {
public:
    bool hasCycle(ListNode *head) {
        ListNode* low = head, *fast = head;
        do {
            if(!fast || !fast->next)
                return false;
            low = low->next;
            fast = fast->next->next;
        }while(low != fast);

        return true;
    }
};

2 删除链表的倒数第 N 个结点

19. 删除链表的倒数第 N 个结点 - 力扣(LeetCode)

快指针先走n步,慢指针后走(k-n步)。

cpp 复制代码
class Solution {
public:
    ListNode* removeNthFromEnd(ListNode* head, int n) {
        // 由于可能会删除链表头部,用哨兵节点简化代码
        ListNode* res = new ListNode(0, head);
        ListNode* low = res, *fast = res;

        while(n--) {
            fast = fast->next;
        }

        while(fast->next) {
            fast = fast->next;
            low = low->next;
        }

        // 左指针的下一个节点就是倒数第 n 个节点
        low->next = low->next->next;

        return res->next; 
    }
};

3 环形链表II

142. 环形链表 II - 力扣(LeetCode)

参考题解:Krahets - 力扣(LeetCode)

使用快慢指针

  • fast 指针走过链表末端,说明链表无环,此时直接返回 null

  • fast == slow时,有环,且两指针在环中第一次相遇。

    将以上两式相减得到s = nb而环的入口的表达式为: a + nb a 即从 head 出发走到环入口的距离(步数), nb 即绕了几圈。又 s = nb ,即慢指针再走 a 步即可到达环入口。

    • fast 走的步数是 slow 步数的 2 倍,即 f = 2s;(解析: fast 每轮走 2 步)

    • fast slow 多走了 n 个环的长度,即 f = s + nb;( 解析: 双指针在环内绕圈直到重合,重合时 fast slow 多走环的长度整数倍 )。

  • fast 重新指向链表头部节点,此时 f = 0s=nb,令slow fast 同时每轮向前走 1 步。当两指针重合时,说明fastslow都满足a + nb。 即fast 指针走到f = aslow 指针走到 s = a + nb

  • 最后返回slowfast即可。

cpp 复制代码
class Solution {
public:
    ListNode *detectCycle(ListNode *head) {
        ListNode* slow = head, *fast = head;
        do {
            if(!fast || !fast->next)
                return NULL;
            fast = fast->next->next;
            slow = slow->next;
        } while(fast != slow);

        // 此时 f = 2s, f = s + nb
        // 所以 s = nb
        // 而环入口节点表达式:a + nb
        
        // 所以让其在入口处汇合:
        fast = head;
        while(slow != fast) {
            fast = fast->next;
            slow = slow->next;
        }
        return slow;
    }
};
相关推荐
测开小菜鸟21 分钟前
使用python向钉钉群聊发送消息
java·python·钉钉
好奇龙猫1 小时前
【学习AI-相关路程-mnist手写数字分类-win-硬件:windows-自我学习AI-实验步骤-全连接神经网络(BPnetwork)-操作流程(3) 】
人工智能·算法
P.H. Infinity1 小时前
【RabbitMQ】04-发送者可靠性
java·rabbitmq·java-rabbitmq
生命几十年3万天1 小时前
java的threadlocal为何内存泄漏
java
sp_fyf_20241 小时前
计算机前沿技术-人工智能算法-大语言模型-最新研究进展-2024-11-01
人工智能·深度学习·神经网络·算法·机器学习·语言模型·数据挖掘
caridle2 小时前
教程:使用 InterBase Express 访问数据库(五):TIBTransaction
java·数据库·express
^velpro^2 小时前
数据库连接池的创建
java·开发语言·数据库
苹果醋32 小时前
Java8->Java19的初步探索
java·运维·spring boot·mysql·nginx
ChoSeitaku2 小时前
链表交集相关算法题|AB链表公共元素生成链表C|AB链表交集存放于A|连续子序列|相交链表求交点位置(C)
数据结构·考研·链表
秋の花2 小时前
【JAVA基础】Java集合基础
java·开发语言·windows