最小生成樹

(Minimum Spanning Tree)

樹?

複習一下,樹是什麼?

一張連通且每個點恰有一條路徑的圖

若有 \(V\) 個點,則有 \(V-1\) 條邊

樹,分為有根樹無根樹

而這棵樹有無邊權皆可

而一棵樹的邊,不會形成環

那生成樹是什麼?

                          (Spanning Tree)

生成樹

 

對於一張圖 \(G=(V,E)\)

我們可以找到有 \(V\) 個點的樹 \(T=(V,E')\),且 \(E \subset E'\)

而對於一張圖,可能有多個不同的生成樹

看定義很難理解,我們來看圖

生成樹

生成樹

而生成樹根據性質有不同分類

最常見的有以下

 

最小生成樹 (Minimum Spanning Tree)

 

DFS 生成樹 (DFS Spanning Tree)

最小生成樹

DFS 生成樹

而我們今天要教的是最小生成樹

 

DFS 生成樹等我們講到圖的連通性時會再提

最小生成樹

最小生成樹的定義

 

對於一張圖的所有生成樹

樹的邊權總和最小的就被我們稱為 最小生成樹

而一張圖可能會有多個不同的 最小生成樹

 

我們一般簡稱 最小生成樹 MST

最小生成樹

最小生成樹的兩種性質

 

Cycle Property

在任何一個圖 \(G\) 上的環

環上邊權最大的邊 \(e\) 一定不在 MST 上

(假設最大邊權不重複)

 

Cut Property

在任何一個圖 \(G\) 上的 Cut

Cut 上最小邊權的邊 \(e\) 一定在 MST 上

(假設最小邊權不重複)

什麼時候 MST 不唯一?

 

Cycle Property

在任何一個圖 \(G\) 上的環

環上邊權最大的邊 \(e\) 一定不在 MST 上

(假設最大邊權不重複)

當環上有多個最大邊權的邊時

Kruskal's Algorithm

運作如下

想法

 

既然我們想要找最小權重的一棵樹

那麼我們每次都把最小的邊加到圖上

 

如果這條邊加入之後,會形成環 \(\Rightarrow\) 不加

如果這條邊加入之後,不會形成環 \(\Rightarrow\)

 

最後是不是就會得到一棵最小權重和的生成樹了呢?

 

想法

 

既然我們想要找最小權重的一棵樹

那麼我們每次都把最小的邊加到圖上

 

如果這條邊加入之後,會形成環 \(\Rightarrow\) 不加

如果這條邊加入之後,不會形成環 \(\Rightarrow\)

 

最後是不是就會得到一棵最小權重和的生成樹了呢?

 

因此,這個演算法也是貪心演算法!

原理

 

為什麼這樣會對呢?

我們這裡就要回到剛剛所講的 Cycle Property

 

 

 

 

Cycle Property

在任何一個圖 \(G\) 上的環

環上邊權最大的邊 \(e\) 一定不在 MST 上

(假設最大邊權不重複)

原理

 

那麼我們只要每次都加最小的邊

出現環不要加

那麼我們一定不會加到環上邊權最大的邊 \(e\)

 

 

 

 

Cycle Property

在任何一個圖 \(G\) 上的環

環上邊權最大的邊 \(e\) 一定不在 MST 上

(假設最大邊權不重複)

但是這裡有個問題,我們要怎麼判斷加入這條邊之後

圖上會不會出現環呢?

 

但是這裡有個問題,我們要怎麼判斷加入這條邊之後

圖上會不會出現環呢?

 

 

沒錯! 只要使用 並查集 來維護即可!

實作步驟

1. 將邊根據邊權進行排序

2. 依序掃過這些邊,能加就加

3. 找到最小生成樹了!

程式碼

1. 將邊根據邊權進行排序

struct edge{
	int w,u,v;
};

vector<edge> edges;

//輸入邊

sort(edges.begin(),edges.end());

程式碼

1. 將邊根據邊權進行排序

struct edge{
	int w,u,v;
};

vector<edge> edges;

//輸入邊

sort(edges.begin(),edges.end());

for(auto e : edges){
	if(find(e.u) != find(e.v)){
		//e.u, e.v 即為最小生成樹的一條邊
    }
}

2. 依序掃過這些邊,能加就加

程式碼

1. 將邊根據邊權進行排序

struct edge{
    int w,u,v;
};

vector<edge> edges;

//輸入邊

sort(edges.begin(),edges.end());

for(auto e : edges){
	if(find(e.u) != find(e.v)){
		//e.u, e.v 即為最小生成樹的一條邊
    }
}

2. 依序掃過這些邊,能加就加

做完了!

時間複雜度?

1. 將邊根據邊權進行排序 \(\Rightarrow O(E \log E)\)

2. 依序掃過這些邊,能加就加 \(\Rightarrow O(E)\)

時間複雜度?

1. 將邊根據邊權進行排序 \(\Rightarrow O(E \log E)\)

2. 依序掃過這些邊,能加就加 \(\Rightarrow O(E)\)

總時間複雜度: \(O(E \log E)\)

或者因為 \(E \le \frac{V(V-1)}{2}\)

所以總時間複雜度也可寫成 \(O(E \log V)\)

Prim's Algorithm

運作如下

想法

既然要找 權重和最小的生成樹

那我們隨便從一個起點開始

每次都走 權重最小的邊

直到所有點都被走過

這樣是不是就會得到 最小生成樹 了呢?

想法

既然要找 權重和最小的生成樹

那我們隨便從一個起點開始

每次都走 權重最小的邊

直到所有點都被走過

這樣是不是就會得到 最小生成樹 了呢?

這也是貪心演算法!

為什麼會對?

為什麼會對?

 

Cut Property

在任何一個圖 \(G\) 上的 Cut

Cut 上最小邊權的邊 \(e\) 一定在 MST 上

(假設最小邊權不重複)

為什麼會對?

 

Cut Property

在任何一個圖 \(G\) 上的 Cut

Cut 上最小邊權的邊 \(e\) 一定在 MST 上

(假設最小邊權不重複)

因此,當我們在做 Prim's Algorithm 時

我們都會從這個連通分量走最小的邊

而這樣,我們就會符合 Cut Property

每次增加的邊皆在 MST

實作步驟

1. 開一個存邊的 priority_queue

2. 任選一個起點開始走

3. 當 pq 不是空的時候,看 pq 頂部的邊是否能走

4. 走到一個點時,將他相鄰的邊加進 pq

5. 走過的所有邊即形成最小生成樹

程式碼

1. 開一個存邊的 priority_queue

priority_queue<pair<int,int>,vector<pair<int,int>>,greater<>> pq;
//邊存的是 {邊權, 走到的點}

程式碼

1. 開一個存邊的 priority_queue

priority_queue<pair<int,int>,vector<pair<int,int>>,greater<>> pq;
//邊存的是 {邊權, 走到的點}

pq.push(1);
visited[1] = true;

2. 任選一個起點開始走

程式碼

1. 開一個存邊的 priority_queue

priority_queue<pair<int,int>,vector<pair<int,int>>,greater<>> pq;
//邊存的是 {邊權, 走到的點}

pq.push(1);
visited[1] = true;

while(!pq.empty()){
    auto [w,u] = pq.top(); pq.pop();
    if(visited[u]) continue;
}

2. 任選一個起點開始走

3. 當 pq 不是空的時候,看 pq 頂部的邊是否能走

程式碼

1. 開一個存邊的 priority_queue

priority_queue<pair<int,int>,vector<pair<int,int>>,greater<>> pq;
//邊存的是 {邊權, 走到的點}

pq.push(1);
visited[1] = true;

while(!pq.empty()){
    auto [ww,u] = pq.top(); pq.pop();
    if(visited[u]) continue;
    
    //如果想要知道是哪個點走到 u 可以額外紀錄
    //但 {from[u], u} 是最小生成樹上的一條邊
    
    for(auto [v,w] : adj[u]){
    	pq.push({w,v});
    }
}

2. 任選一個起點開始走

3. 當 pq 不是空的時候,看 pq 頂部的邊是否能走

4. 走到一個點時,將他相鄰的邊加進 pq

時間複雜度

 

 

有一個存邊的 priority_queue

而且跑 \(E\) 次

 

\(O(E \log E) \Rightarrow O(E \log V)\) 

 

如果不用 priority_queue

時間複雜度可以達到 \(O(V^2)\)

完全圖 時快很多

例題

有 \(n\) 個國家,而今天每個國家都有石油的需求

而今天有 \(m\) 種不同的道路可以建造,每個道路建造費用 \(w_i\)

這些國家也可以選擇自己花費 \(c_i\) 開採石油

請問讓每個國家取得石油的最少費用是多少?

這題看起來可以直接做最小生成樹

不過有 每個國家能得到石油 的限制

因此我們必須要想個方式來做到這一點

這題看起來可以直接做最小生成樹

不過有 每個國家能得到石油 的限制

因此我們必須要想個方式來做到這一點

建設虛點!

建設虛點!

將開採石油井這件事開成一個點

以 \(c_i\) 為邊權連到每個國家

跑最小生成樹演算法!

完成!

YTP 2021 決賽 p4

 

今天有 \(n\) 個國家,他們之間有 \(m\) 條邊

邊權為 \(w_i\),而從國家 \(u\) 派人到國家 \(v\) 的花費

為從 \(u\) 走到 \(v\) 的路徑上最大的邊

今天你想知道每個國家互相走到對方的最少花費

Kruskal 演算法和 並查集 維護連通塊大小

然後把答案加上 \(w_i \times cnt[find(u)] \times cnt[find(v)]\)

練習題

TIOJ 2164 - 運送蛋餅 (這題是完全圖 用Prim's)

 

NPSC 2018 高中組決賽 pB - 平衡的技能樹

(這題的技巧大家不會,但是可以想想看大概做法)

 

IOI 2003 - Trail Maintenance

最小生成樹

By sam571128

最小生成樹

  • 121