用 龜兔賽跑演算法 來檢查有沒有環路Linked List Cycle_Leetcode #141

更新 發佈閱讀 5 分鐘

題目敘述

給定一個Linked list鏈結串列的Head node,

請判斷這條Linked list是否存在環路(Cycle)?

如果有環路,回傳True。

如果沒有,回傳False。


題目的原文敘述


測試範例

Example 1:

raw-image


Input: head = [3,2,0,-4], pos = 1
Output: true
Explanation: There is a cycle in the linked list, where the tail connects to the 1st node (0-indexed).
有環路Cycle​


Example 2:

raw-image


Input: head = [1,2], pos = 0
Output: true
Explanation: There is a cycle in the linked list, where the tail connects to the 0th node.
有環路Cycle


Example 3:

raw-image


Input: head = [1], pos = -1
Output: false
Explanation: There is no cycle in the linked list.
沒有環路 

約束條件

Constraints:

  • The number of the nodes in the list is in the range [0, 10^4].

Linked list的節點總數目介於 0 ~ 一萬之間。

  • -10^5 <= Node.val <= 10^5

每個節點的節點值都介於 負十萬~正十萬 之間。

  • pos is -1 or a valid index in the linked-list.

pos代表發生環路的節點編號,如果是-1代表沒有環路。


進階提問

Follow up: Can you solve it using O(1) (i.e. constant) memory?

能不能在常數空間複雜度的條件下,解出這題?


演算法 雙指針(快/慢 指針) 俗稱龜兔賽跑演算法

有一個比較直覺的想法,就是沿路記下每個節點的ID,在C/C++裡面可以記錄節點的記憶體位置,在python裡面可以記錄run-time的物件ID。如果在拜訪的過程中,遇到相同的ID,就代表有環路。

但是,這個方法會耗費O(n)的空間成本去紀錄節點的ID,有沒有更好的方法呢?

其實有喔!


(俗稱龜兔賽跑演算法,或者Floyd cycle detection algorithm)

想一下,假如有環路發生,我們可以說linked list中的某一塊形成了封閉的環型,就好像操場跑道的形狀

我們讓兩個跑者(在演算法裡面就是雙指針,快/慢指針),同時從起點出發,快的固定都跑得比慢的還快,最後快的一定可以領先一圈,又遇到慢的跑者




相反的,如果沒有環路,相當於linked list是一條直線型的跑道,那麼我們讓兩個跑者同時從起點出發,兩者並不會在中途相遇


為了方便實作,我們可以假設快跑者是慢跑者的兩倍速度,讓他們同時從起點出發

假如快、慢跑者中途會相遇,那麼代表linked list 存在至少一個環路
假如快、慢跑者不會相遇,那麼代表linked list是一條直線的串列,沒有環路

這樣我們只要維護兩個pointer 快、慢指針就相當於快、慢跑者,
只需要常數空間O(1)即可。


程式碼 雙指針(快/慢 指針) 俗稱龜兔賽跑演算法

class Solution:
def hasCycle(self, head: ListNode) -> bool:

# two pointers
slow, fast = head, head

# slow runner move 1 step, fast runner move 2 steps per iteration
# if fast runner meets slow one somewhere, then there is a cycle in linked list

while fast and fast.next:

slow, fast = slow.next, fast.next.next

if slow is fast:
return True

return False

複雜度分析

時間複雜度:

需要掃描過每個節點,所需時間為O(n)。

空間複雜度:

只需要維護固定尺寸的臨時變數,所需空間為O(1) 常數級別。


關鍵知識點

假如有環路發生,我們可以說linked list中的某一塊形成了封閉的環型,就好像操場跑道的形狀

我們讓兩個跑者(在演算法裡面就是雙指針,快/慢指針),同時從起點出發,快的固定都跑得比慢的還快,最後快的一定可以領先一圈,又遇到慢的跑者


相反的,如果沒有環路,相當於linked list是一條直線型的跑道,那麼我們讓兩個跑者同時從起點出發,兩者並不會在中途相遇


Reference:

[1] Linked List Cycle - LeetCode

留言
avatar-img
留言分享你的想法!
avatar-img
小松鼠的演算法樂園
97會員
428內容數
由有業界實戰經驗的演算法工程師, 手把手教你建立解題的框架, 一步步寫出高效、清晰易懂的解題答案。 著重在讓讀者啟發思考、理解演算法,熟悉常見的演算法模板。 深入淺出地介紹題目背後所使用的演算法意義,融會貫通演算法與資料結構的應用。 在幾個經典的題目融入一道題目的多種解法,或者同一招解不同的題目,擴展廣度,並加深印象。
2024/09/10
Insert Greatest Common Divisors in Linked List 題目給定一個鏈結串列, 請在兩兩節點之間加入一個新節點,新節點的值為兩者之間的最大公因數。 最後返回新串列的head node作為答案。
Thumbnail
2024/09/10
Insert Greatest Common Divisors in Linked List 題目給定一個鏈結串列, 請在兩兩節點之間加入一個新節點,新節點的值為兩者之間的最大公因數。 最後返回新串列的head node作為答案。
Thumbnail
2024/05/07
題目敘述 輸入給定一個鏈結串列,整體看代表一個十進位的數字,各別看每個節點代表每個digit,分別從最高位~最低位個位數。 要求我們把原本的數字乘以二,並且以鏈結串列的形式返回答案。 原本的英文題目敘述
Thumbnail
2024/05/07
題目敘述 輸入給定一個鏈結串列,整體看代表一個十進位的數字,各別看每個節點代表每個digit,分別從最高位~最低位個位數。 要求我們把原本的數字乘以二,並且以鏈結串列的形式返回答案。 原本的英文題目敘述
Thumbnail
2024/05/06
題目敘述 題目給定一個鏈結串列中的節點Node,要求我們從鏈結串列中刪除該節點。 題目保證該節點不是tail node。 題目要求我們in-place原位操作。 原本的英文題目敘述 測試範例 Example 1: Input: head = [4,5,1,9], node = 5
Thumbnail
2024/05/06
題目敘述 題目給定一個鏈結串列中的節點Node,要求我們從鏈結串列中刪除該節點。 題目保證該節點不是tail node。 題目要求我們in-place原位操作。 原本的英文題目敘述 測試範例 Example 1: Input: head = [4,5,1,9], node = 5
Thumbnail
看更多
你可能也想看
Thumbnail
不是每個人都適合自己操盤,懂得利用「專業」,才是績效拉開差距的開始
Thumbnail
不是每個人都適合自己操盤,懂得利用「專業」,才是績效拉開差距的開始
Thumbnail
給定一個輸入陣列,每一個tuple代表節點之間了從屬關係。 請從從屬關係重建整顆二元樹,並且返回整顆二元樹的根結點。
Thumbnail
給定一個輸入陣列,每一個tuple代表節點之間了從屬關係。 請從從屬關係重建整顆二元樹,並且返回整顆二元樹的根結點。
Thumbnail
題目敘述 Merge Nodes in Between Zeros 給定一個鏈結串列,合併非零區間的節點(以加總的方式合併),輸出合併後的鏈結串列。
Thumbnail
題目敘述 Merge Nodes in Between Zeros 給定一個鏈結串列,合併非零區間的節點(以加總的方式合併),輸出合併後的鏈結串列。
Thumbnail
題目敘述 All Ancestors of a Node in a Directed Acyclic Graph 給定一個有向無環圖,請找出每個點的祖先,以陣列的形式返回答案。
Thumbnail
題目敘述 All Ancestors of a Node in a Directed Acyclic Graph 給定一個有向無環圖,請找出每個點的祖先,以陣列的形式返回答案。
Thumbnail
題目敘述 題目給定一個鏈結串列中的節點Node,要求我們從鏈結串列中刪除該節點。 題目保證該節點不是tail node。 題目要求我們in-place原位操作。 原本的英文題目敘述 測試範例 Example 1: Input: head = [4,5,1,9], node = 5
Thumbnail
題目敘述 題目給定一個鏈結串列中的節點Node,要求我們從鏈結串列中刪除該節點。 題目保證該節點不是tail node。 題目要求我們in-place原位操作。 原本的英文題目敘述 測試範例 Example 1: Input: head = [4,5,1,9], node = 5
Thumbnail
題目敘述 輸入給定一個鏈結串列的head node。 要求我們進行化簡,只要某個節點的右手邊存在比較大的節點,就刪除掉。 例如 5->2->13->3 5的右手邊有13,所以5刪除掉。 2的右手邊有13,所以2刪除掉。 13的右手邊沒有更大的節點,所以13留著。 3的右手邊沒有更大
Thumbnail
題目敘述 輸入給定一個鏈結串列的head node。 要求我們進行化簡,只要某個節點的右手邊存在比較大的節點,就刪除掉。 例如 5->2->13->3 5的右手邊有13,所以5刪除掉。 2的右手邊有13,所以2刪除掉。 13的右手邊沒有更大的節點,所以13留著。 3的右手邊沒有更大
Thumbnail
給定一個Linked list鏈結串列的Head node, 請判斷這條Linked list是否存在環路(Cycle)? 如果有環路,回傳True。 如果沒有,回傳False。
Thumbnail
給定一個Linked list鏈結串列的Head node, 請判斷這條Linked list是否存在環路(Cycle)? 如果有環路,回傳True。 如果沒有,回傳False。
Thumbnail
Node 對我來說一直是很似懂非懂的概念,因為沒有實際可見的數字或字串實物可以觀察中間步驟,比起其他解題抽象許多。以下解題方法來自網友與 GPT 提供的答案,我想在本文中嘗試自己做視覺化圖解,說明 CodeWars - Can you get the loop ? 的解題思路。
Thumbnail
Node 對我來說一直是很似懂非懂的概念,因為沒有實際可見的數字或字串實物可以觀察中間步驟,比起其他解題抽象許多。以下解題方法來自網友與 GPT 提供的答案,我想在本文中嘗試自己做視覺化圖解,說明 CodeWars - Can you get the loop ? 的解題思路。
追蹤感興趣的內容從 Google News 追蹤更多 vocus 的最新精選內容追蹤 Google News