题面
题目描述
Bob喜欢玩电脑游戏,特别是战略游戏。但是他经常无法找到快速玩过游戏的办法。现在他有个问题。他要建立一个古城堡,城堡中的路形成一棵树。他要在这棵树的结点上放置最少数目的士兵,使得这些士兵能了望到所有的路。注意,某个士兵在一个结点上时,与该结点相连的所有边将都可以被了望到。
请你编一程序,给定一树,帮Bob计算出他需要放置最少的士兵。
输入格式
输入文件中数据表示一棵树,描述如下: 第一行 N,表示树中结点的数目。 第二行至第N+1行,每行描述每个结点信息,依次为:该结点标号i,k(后面有k条边与结点I相连),接下来k个数,分别是每条边的另一个结点标号r1,r2,…,rk。 对于一个
n(0<n<=1500)
n
(
0
<
n
<=
1500
)
个结点的树,结点标号在0到n-1之间,在输入文件中每条边只出现一次。
输出格式
输出文件仅包含一个数,为所求的最少的士兵数目。
题解
树形DP裸题。
f[i][1]
f
[
i
]
[
1
]
代表第i个结点,选择放士兵的最小答案。
f[i][0]
f
[
i
]
[
0
]
代表第i个结点,选择不放士兵的最小答案。
状态转移方程:
code
#include<bits/stdc++.h>
using namespace std;
inline int read(){
int num=0;
char c=' ';
bool flag=true;
for(;c>'9'||c<'0';c=getchar())
if(c=='-')
flag=false;
for(;c>='0'&&c<='9';num=num*10+c-48,c=getchar());
return flag ? num : -num;
}
namespace graph{
const int maxn=1520;
vector<int>G[maxn];
bool in[maxn];
int n;
void init(){
n=read();
for(int i=1;i<=n;i++){
int dot=read();
int num=read();
for(int j=1;j<=num;j++){
int x=read();
G[dot].push_back(x);
in[x]=true;
}
}
}
}using namespace graph;
int f[maxn][2];
void dp(int u,int fa){
int size=G[u].size();
int t0=0,t1=0;
for(int i=0;i<size;i++){
int v=G[u][i];
if(v==fa)continue;
dp(v,u);
t0+=f[v][1];
t1+=min(f[v][1],f[v][0]);
}
f[u][0]=t0;
f[u][1]=t1+1;
}
int main(){
init();
int mark=0;
for(int i=0;i<n;i++)
if(!in[i]){
mark=i;
break;
}//寻找没有入度的根节点。
dp(mark,-1);
printf("%d\n",min(f[mark][0],f[mark][1]));
return 0;
}