题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4003

思路:dp[i][j]表示以i为根选择j个机器人的最小花费,然后就是背包了:dp[u][i]=min(dp[u][i],dp[u][i-j]+dp[v][j]+j*w)(1<<j<<i,u是v的根)。值得注意的是dp[u][0]的意义,dp[i][0]表示以i为根的子树派一个机器人遍历完整棵子树后又回到i的花费。

hdu 4003(树形dp)_我的dp之路hdu 4003(树形dp)_i++_02
 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<algorithm>
 5 #include<vector>
 6 using namespace std;
 7 #define MAXN 10100
 8 #define inf 1<<30
 9 
10 struct Edge{
11     int v,w;
12     Edge(int vv,int ww):v(vv),w(ww){}
13 };
14 
15 int n,s,k;
16 int dp[MAXN][20];
17 vector<vector<Edge> >G;
18 
19 void dfs(int u,int father)
20 {
21     for(int i=0;i<G[u].size();i++){
22         int v=G[u][i].v,w=G[u][i].w;
23         if(v==father)continue;
24         dfs(v,u);
25         for(int i=k;i>=0;i--){
26             dp[u][i]+=dp[v][0]+2*w;//v所在子树不放置机器人,那么必然要派一个机器人从u到v去遍历v所在子树并且回到u.
27             for(int j=1;j<=i;j++){ //子树放置1-j个机器人
28                 dp[u][i]=min(dp[u][i],dp[u][i-j]+dp[v][j]+j*w);
29             }
30         }
31     }
32 }
33 
34 int main()
35 {
36     int u,v,w;
37     while(~scanf("%d%d%d",&n,&s,&k)){
38         G.clear();
39         G.resize(n+2);
40         for(int i=1;i<n;i++){
41             scanf("%d%d%d",&u,&v,&w);
42             G[u].push_back(Edge(v,w));
43             G[v].push_back(Edge(u,w));
44         }
45         memset(dp,0,sizeof(dp));
46         dfs(s,-1);
47         printf("%d\n",dp[s][k]);
48     }
49     return 0;
50 }
View Code