經典圖論題 Amount of Time for Binary Tree to Be Infected_2385

閱讀時間約 6 分鐘

題目敘述

題目會給定我們一棵二元數Binary Tree的根結點。

並且給定感染的病毒源節點位置,每個單位時間,可以向相鄰的節點感染一次,問我們需要多少時間去感染整棵樹?

題目的原文敘述


測試範例

Example 1:

raw-image
Input: root = [1,5,3,null,4,10,6,9,2], start = 3
Output: 4
Explanation: The following nodes are infected during:
- Minute 0: Node 3
- Minute 1: Nodes 1, 10 and 6
- Minute 2: Node 5
- Minute 3: Node 4
- Minute 4: Nodes 9 and 2
It takes 4 minutes for the whole tree to be infected so we return 4.

Example 2:

raw-image
Input: root = [1], start = 1
Output: 0
Explanation: At minute 0, the only node in the tree is infected so we return 0.

約束條件

Constraints:

  • The number of nodes in the tree is in the range [1, 10^5].
  • 1 <= Node.val <= 10^5
  • Each node has a unique value.
  • A node with a value of start exists in the tree.

演算法

其實,二元樹Binary tree本身也是一種圖Graph,轉換成圖之後,要解開就非常容易了。

因為,最少需要多少時間去感染整棵樹 = 最少需要多少時間,去拜訪整張圖。

我們就把原本感染的問題映射到圖上的拜訪Traversal問題去解開來。


第一步:

先用DFS深度優先演算法建立整張圖,也就是把二元樹的連結關係相鄰陣列的方式來表達

第二步:

再用BFS廣度優先演算法,從感染的起點開始,拜訪整張圖,所需的時間就是感染整顆樹的時間。(BFS先天具有點波源擴散的性質)


程式碼

class Solution: 		
def amountOfTime(self, root: Optional[TreeNode], start: int) -> int:
graph = defaultdict(list)

# Build connected graph for binary tree in DFS
def dfs(node, parent):

if not node:
return

if parent:
graph[node.val].append(parent.val)
graph[parent.val].append(node.val)

dfs(node.left, node)
dfs(node.right, node)
return
#--------------------------------------
dfs(root, None)

# Start infection from start node, with BFS, which naturally has shortest path
bfs_queue = deque( [ (start, 0) ] )
visited = set([start])

while bfs_queue:

for _ in range( len(bfs_queue) ):
cur, step = bfs_queue.popleft()

for neighbor in graph[cur]:
if neighbor not in visited:
visited.add(neighbor)
bfs_queue.append( (neighbor, step+1) )

return step


關鍵知識點

二元樹Binary tree本身也是一種圖Graph,轉換成圖之後,要解開就非常容易了。

因為,最少需要多少時間去感染整棵樹 = 最少需要多少時間,去拜訪整張圖。


另外,BFS先天具有點波源擴散的性質,所以在無權重的圖中,也具有尋找最短路徑的特質。在這題,就是利用這個特質,來找出最少需要多少時間來拜訪整張圖(相當於感染整棵樹)


複雜度分析

時間複雜度:

O(n) DFS和BFS拜訪整張圖,每個節點至多分別拜訪一次,O(2n) = O(n)。

空間複雜度:

O(n) DFS和BFS拜訪整張圖,最大深度發生在整棵樹向左歪斜或者向右歪斜時 = O(n)。

另外,建立graph的相連矩陣時,也需要耗費O(n)的空間。


Reference:

[1] Python by DFS + BFS [w/ Comment]

avatar-img
90會員
425內容數
由有業界實戰經驗的演算法工程師, 手把手教你建立解題的框架, 一步步寫出高效、清晰易懂的解題答案。 著重在讓讀者啟發思考、理解演算法,熟悉常見的演算法模板。 深入淺出地介紹題目背後所使用的演算法意義,融會貫通演算法與資料結構的應用。 在幾個經典的題目融入一道題目的多種解法,或者同一招解不同的題目,擴展廣度,並加深印象。
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
題目敘述 題目會給我們一個輸入陣列prerequisites,每個pair代表兩個課程之間的先修關係,和課程總數numCourses。 題目問我們這組課程表是否能依照順序修完所有的課程? 如果可以,返回True。 如果不行,代表有擋修形成死結,無法依照順序修完所有的課程,返回False。
題目敘述 題目會給定我們一顆二元搜索樹的根結點root,和任意兩個樹中的節點p和q。 要求我們找出p, q最靠近的公共祖先節點。 題目的原文敘述 測試範例 Example 1: Input: root = [6,2,8,0,4,7,9,null,null,3,5], p = 2, q
題目敘述 題目會給我們一個不規則排列的二維陣列,要求我們列出從起點出發,走次對角線,由左下到右上逐層拜訪的路徑。
題目會給我們一個routes 陣列,裡面都是分別代表每一條公車路線所對應的公車站編號。 題目要求我們計算出,從起點站source到終點站target的最精簡公車路線搭乘次數是幾次? 也就是說,就是在最少轉乘的前提下,旅途中需要搭乘幾條公車路線?
題目會給我們一個pair陣列,裡面都是原本陣列相鄰元素形成的pair,順序已經被打散。 要求我們從pair陣列重建出原本的陣列。 答案可能有不只一組,任選一組回傳即可。
題目會給定我們兩個點座標,分別是起點和終點。 另外,還有一個參數t,代表時間限制。 從起點出發之後,每一秒鐘,我們必須選擇一個N8 8連通的方向,往鄰居的格子點移動。(題目有特別強調,每一秒必須強制移動到下一個格子點,不能停留在原地) 請問我們能不能在時間限制內,從起點走到終點?
題目敘述 題目會給我們一個輸入陣列prerequisites,每個pair代表兩個課程之間的先修關係,和課程總數numCourses。 題目問我們這組課程表是否能依照順序修完所有的課程? 如果可以,返回True。 如果不行,代表有擋修形成死結,無法依照順序修完所有的課程,返回False。
題目敘述 題目會給定我們一顆二元搜索樹的根結點root,和任意兩個樹中的節點p和q。 要求我們找出p, q最靠近的公共祖先節點。 題目的原文敘述 測試範例 Example 1: Input: root = [6,2,8,0,4,7,9,null,null,3,5], p = 2, q
題目敘述 題目會給我們一個不規則排列的二維陣列,要求我們列出從起點出發,走次對角線,由左下到右上逐層拜訪的路徑。
題目會給我們一個routes 陣列,裡面都是分別代表每一條公車路線所對應的公車站編號。 題目要求我們計算出,從起點站source到終點站target的最精簡公車路線搭乘次數是幾次? 也就是說,就是在最少轉乘的前提下,旅途中需要搭乘幾條公車路線?
題目會給我們一個pair陣列,裡面都是原本陣列相鄰元素形成的pair,順序已經被打散。 要求我們從pair陣列重建出原本的陣列。 答案可能有不只一組,任選一組回傳即可。
題目會給定我們兩個點座標,分別是起點和終點。 另外,還有一個參數t,代表時間限制。 從起點出發之後,每一秒鐘,我們必須選擇一個N8 8連通的方向,往鄰居的格子點移動。(題目有特別強調,每一秒必須強制移動到下一個格子點,不能停留在原地) 請問我們能不能在時間限制內,從起點走到終點?
你可能也想看
Google News 追蹤
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
前言 如文章的封面圖片1/2~1/27總共有幾天呢? 如果你的答案是25天,那務必把這個日期的地雷搞懂 種樹問題計算 題目 一條路有50公尺,每10公尺種一棵樹,請問總共種了幾顆樹? 這時就會直覺,阿不就是50/10=5,總共種了5顆樹!! 但其實大錯特錯哦❌
Thumbnail
可能包含敏感內容
大樹
Thumbnail
緩慢沒有關係,一旦植入種子,總有一天就會找到發芽的路徑,然後用自己的方式開心茁壯。🌱
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
前言 如文章的封面圖片1/2~1/27總共有幾天呢? 如果你的答案是25天,那務必把這個日期的地雷搞懂 種樹問題計算 題目 一條路有50公尺,每10公尺種一棵樹,請問總共種了幾顆樹? 這時就會直覺,阿不就是50/10=5,總共種了5顆樹!! 但其實大錯特錯哦❌
Thumbnail
可能包含敏感內容
大樹
Thumbnail
緩慢沒有關係,一旦植入種子,總有一天就會找到發芽的路徑,然後用自己的方式開心茁壯。🌱