快慢指针法应用合集

在链表题目中,为了降低时间复杂度和空间复杂度,我们有的时候会使用快慢指针法,力求对链表遍历一次就实现想要的功能。

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

参考题目

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

解题思路

总体目标

快指针先走几步,然后快慢指针一起走,最终快指针指向null的时候,慢指针正好指向要删除结点的前一个结点,借慢指针来实现倒数第N个结点的删除。

实现方法

创建一个虚拟头结点,并使其指向当前链表的头结点。

slow指针从虚拟头结点dummyhead出发,fast指针从头结点head出发。(如果都从head出发,则实现的是slow指针最后指向要删除的结点。正如前面所说,为了便于删除,要使得慢指针正好指向要删除结点的前一个结点。因此,slow从dummyhead出发,fast从head出发)

开始时,fast指针先行N步,slow指针原地待命。

然后,fast指针和slow指针同时向后迈步,每次迈一步,直至fast指针指向NULL。此时,slow指针指向的就正好是要删除结点的前一个结点。在此进行删除工作即可。

注意

本题中确保了N不会大于链表长度,因此也就不需要考虑要删除的结点不存在的问题。

否则,应将N与链表长度作比较,确定倒数第N个链表结点存在后,再进行上述的删除操作。

参考代码

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* removeNthFromEnd(ListNode* head, int n) {
        ListNode* dummyhead = new ListNode(0, head);
        ListNode* fast = head;
        ListNode* slow = dummyhead;
        for (int i = 0; i < n; ++i) {
            fast = fast->next;
        }
        while (fast!=nullptr) {
            fast = fast->next;
            slow = slow->next;
        }
        ListNode* d = slow->next;
        slow->next = slow->next->next;
        ListNode* ans = dummyhead->next;
        delete d;
        delete dummyhead;
        return ans;
    }
};

判断链表中是否有环

参考题目

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

解题思路

设想一个情景:两个人在赛跑,A速度快,B速度慢,若是存在环(勺状图),A和B总是会相遇的,相遇时A所经过的路径的长度要比B多若干个环的长度。

在本题中,快慢指针的思想即是如此,快指针每次走两步,慢指针每次走一步,它们同时从头节点出发。如果二者相遇,则说明存在环;如果快指针到达链表尾NULL,则说明不存在环。

参考代码

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    bool hasCycle(ListNode *head) {
        ListNode* fast = head;
        ListNode* slow = head;
        while(fast!=NULL && fast->next!=NULL){
            fast = fast->next->next;
            slow = slow->next;
            if(slow==fast) return true;
        }
        return false;
    }
};

链表的中间结点

参考题目

876. 链表的中间结点 - 力扣(LeetCode)

解题思路

快慢指针同时从头节点开始,快指针每次走两步,慢指针每次走一步。当快指针走到链表末尾的时候,慢指针正好到达链表的中间结点。(注意特况特判,即链表中有0或1个结点时)

参考代码

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* middleNode(ListNode* head) {
        if(head==nullptr){return nullptr;}
        if(head->next==nullptr) return head;
        ListNode* fast = head->next->next;
        ListNode* slow = head->next;
        while(fast!=nullptr&&fast->next!=nullptr){
            slow = slow->next;
            fast = fast->next->next;
        }
        return slow;
    }
};

判断链表中是否有环并找到环的起点

参考题目

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

解题思路

如果链表中有环,那么说明快慢指针会在环中的某位置相遇。此时,应让二者分别从head和相遇节点同时开始每次走一步。当它们再次相遇时,二者所在的结点便是环的入口结点。

证明:

证明来源:Floyd判圈法(Floyd Cycle Detection Algorithm)-CSDN博客

参考代码

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    ListNode *detectCycle(ListNode *head) {
        ListNode* fast = head;
        ListNode* slow = head;
        while(fast!=NULL && fast->next!=NULL){
            fast = fast->next->next;
            slow = slow->next;
            if(slow == fast){
                ListNode* index1 = fast;
                ListNode* index2 = head;
                while(index1 != index2){
                    index1=index1->next;
                    index2=index2->next;
                }
                return index2;
            }
        }
        return NULL;
    }
};

重排链表

参考题目

143. 重排链表 - 力扣(LeetCode)

解题思路

  1. 快慢指针找到中间结点
  2. 后半条链反转链表
  3. 合并两条链表

参考代码

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* middleNode(ListNode* head){
        if(head==nullptr) return nullptr;
        if(head->next == nullptr) return head;
        ListNode* fast = head->next->next;
        ListNode* slow = head->next;
        while(fast!=nullptr && fast->next!=nullptr){
            slow = slow->next;
            fast = fast->next->next;
        }
        return slow;
    }

    ListNode* reverseList(ListNode* head){
        if(head==nullptr) return nullptr;
        if(head->next == nullptr) return head;
        ListNode* pre = nullptr;
        ListNode* cur = head;
        while(cur!=nullptr){
            ListNode* tmp = cur->next;
            cur->next = pre;
            pre = cur;
            cur = tmp;
        }
        return pre;
    }

    void mergeList(ListNode* head1,ListNode* head2){
        ListNode* tmp1,* tmp2;
        while(head1!=nullptr && head2!=nullptr){
            tmp1 = head1->next;
            tmp2 = head2->next;

            head1->next = head2;
            head1 = tmp1;

            head2->next = head1;
            head2 = tmp2;
        }
    }


    void reorderList(ListNode* head) {
        if(head==nullptr) return;
        ListNode* mid = middleNode(head);
        ListNode* h1 = head;
        ListNode* h2 = mid->next;
        mid->next = nullptr;
        h2 = reverseList(h2);
        mergeList(h1,h2);
    }
};

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值