開啟章節選單

1292 Strategic game

題目說明

給一個有n個節點的樹狀圖,可以在每個節點選擇要不要放置士兵,要求是每一個路徑都要有士兵看見(也就是每一條路徑的兩端點都至少要有一個士兵) 求最少須要放多少士兵

解題思路

這是最小支配集 的問題,用樹形dp解決。 dp[i][0]=在節點i不放士兵的最小花費 dp[i][1]=在節點i放士兵的最小花費 而 dp[i][0]=所有i的子節點e1,e2,...,en,dp[e1][1]加到dp[en][1] 因為當目前節點沒有的話,那每個子節點必須要有士兵,才能符合"任何路徑的兩端至少有一端有士兵" dp[i][1]=對於每個子節點e,取min(dp[e][0],dp[e][1])然後總和 即可得出完整樹形dp式: dp[i][0]=sum of {dp[e][1]},when e contains to i dp[i][1]=sum of {min(dp[e][0],dp[e][1])},when e contains to i

完整程式碼

#include<iostream>
#include<vector>
using namespace std;
static vector<vector<int>> g;
static vector<vector<int>> dp;
void dfs(int u, int fa) {
    dp[u][0] = 0;
    dp[u][1] = 1;
    for (int v : g[u]) {
        if (v == fa)
            continue;
        dfs(v, u);
        dp[u][0] += dp[v][1];
        dp[u][1] += min(dp[v][0], dp[v][1]);
    }
}
int main() {
    int n, u, d, v;
    while (cin >> n && n) {
        g.clear();
        g.resize(n);
        dp.clear();
        dp.resize(n, vector<int>(2, 0));
        for (int i=0; i<n; i++) {
            scanf("%d:(%d)", &u, &d);
            int v;
            for (int j=0; j<d; j++) {
                cin >> v;
                g[u].push_back(v);
                g[v].push_back(u);
            }
        }
        dfs(0, -1);
        cout << min(dp[0][0], dp[0][1]) << endl;
    }
    return 0;
}