【题解】
先左耳子右兄弟转二叉树,方便转移
f[i][j][k]表示在原树中i和i的兄弟节点所在的子树(相当于转换后i及其子树中),在原树中离i最近的父亲点为j,
放置k个工厂的最小距离
转移:
f[i][j][k] = min{
选 f[l[i]][i][a] + f[r[i]][j][k - a - 1]
不选 d[l[i]][j][a] + f[r[i]][j][k - a] + dist[i, j]
}
1 #include <iostream> 2 #include <cstdio> 3 #include <cstdlib> 4 #include <cstring> 5 #include <cmath> 6 #include <algorithm> 7 #define min(a, b) ((a) < (b) ? (a) : (b)) 8 #define max(a, b) ((a) > (b) ? (a) : (b)) 9 10 inline void read(int &x) 11 { 12 x = 0;char ch = getchar(), c = ch; 13 while(ch < ‘0‘ || ch > ‘9‘)c = ch, ch = getchar(); 14 while(ch <= ‘9‘ && ch >= ‘0‘)x = x * 10 + ch - ‘0‘, ch = getchar(); 15 if(c == ‘-‘)x = -x; 16 } 17 18 const int MAXN = 100 + 10; 19 const int MAXK = 50 + 10; 20 int dp[MAXN][MAXN][MAXK],n,m,cost[MAXN],fa[MAXN],dist[MAXN]; 21 int l[MAXN], r[MAXN], flag; 22 23 void dfs(int i) 24 { 25 if(!i && flag)return; 26 flag = 1; 27 dfs(l[i]); 28 dfs(r[i]); 29 int len = dist[i]; 30 for(register int j = fa[i];j != -1;j = fa[j]) 31 { 32 for(register int k = 0;k <= m;++ k) 33 { 34 for(register int a = 0;a <= k;++ a) 35 { 36 if(k - a - 1 >= 0) 37 dp[i][j][k] = min(dp[i][j][k], dp[l[i]][i][a] + dp[r[i]][j][k - a - 1]); 38 if(k - a >= 0) 39 dp[i][j][k] = min(dp[i][j][k], dp[l[i]][j][a] + dp[r[i]][j][k - a] + len * cost[i]); 40 } 41 } 42 len += dist[j]; 43 } 44 } 45 46 int main() 47 { 48 read(n), read(m); 49 for(register int i = 1;i <= n;++ i) 50 { 51 read(cost[i]), read(fa[i]),read(dist[i]); 52 r[i] = l[fa[i]]; 53 l[fa[i]] = i; 54 } 55 memset(dp, 0x3f, sizeof(dp)); 56 memset(dp[0], 0, sizeof(dp[0])); 57 fa[0] = -1; 58 dfs(0); 59 printf("%d", dp[l[0]][0][m]); 60 return 0; 61 }
原文:http://www.cnblogs.com/huibixiaoxing/p/7502107.html