【BZOJ1217】[HNOI2003]消防局的设立 树形DP

【BZOJ1217】[HNOI2003]消防局的设立

Description

2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地。起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两个基地都能够通过道路到达,所以所有的基地形成了一个巨大的树状结构。如果基地A到基地B至少要经过d条道路的话,我们称基地A到基地B的距离为d。由于火星上非常干燥,经常引发火灾,人类决定在火星上修建若干个消防局。消防局只能修建在基地里,每个消防局有能力扑灭与它距离不超过2的基地的火灾。
你的任务是计算至少要修建多少个消防局才能够确保火星上所有的基地在发生火灾时,消防队有能力及时扑灭火灾。

Input

第一行为n,表示火星上基地的数目。N<=1000
接下来的n-1行每行有一个正整数,其中文件第i行的正整数为a[i],表示从编号为i的基地到编号为a[i]的基地之间有一条道路,为了更加简洁的描述树状结构的基地群,有a[i] < i

Output

仅有一个正整数,表示至少要设立多少个消防局才有能力及时扑灭任何基地发生的火灾。

Sample Input

6
1
2
3
4
5

Sample Output

2

题解:侦查守卫的弱化版。用f[i][j]表示i的子树中所有点已被覆盖,且还能向上覆盖j层的最小花费,g[i][j]表示i的j层下面的所有点已被覆盖,还需要覆盖上面的j层的最小花费。转移比较容易。

#include <cstring>
#include <cstdio>
#include <iostream>

using namespace std;
int n,cnt;
const int maxn=100010;
int f[maxn][3],g[maxn][2],to[maxn],next[maxn],head[maxn];
void dfs(int x)
{
	f[x][0]=f[x][1]=f[x][2]=1,g[x][0]=g[x][1]=0;
	for(int y,i=head[x];i!=-1;i=next[i])
	{
		y=to[i],dfs(y);
		f[x][0]=min(f[x][0]+f[y][0],g[x][0]+f[y][1]);
		f[x][1]=min(f[x][1]+g[y][0],g[x][1]+f[y][2]);
		f[x][2]=f[x][2]+g[y][1];
		g[x][0]+=f[y][0];
		g[x][1]+=g[y][0];
		f[x][1]=min(f[x][1],f[x][2]),f[x][0]=min(f[x][0],f[x][1]);
		g[x][0]=min(g[x][0],f[x][0]),g[x][1]=min(g[x][1],g[x][0]);
	}
}
inline int rd()
{
	int ret=0,f=1;	char gc=getchar();
	while(gc<'0'||gc>'9')	{if(gc=='-')	f=-f;	gc=getchar();}
	while(gc>='0'&&gc<='9')	ret=ret*10+(gc^'0'),gc=getchar();
	return ret*f;
}
inline void add(int a,int b)
{
	to[cnt]=b,next[cnt]=head[a],head[a]=cnt++;
}
int main()
{
	n=rd();
	int i,a;
	memset(head,-1,sizeof(head));
	for(i=2;i<=n;i++)	a=rd(),add(a,i);
	dfs(1);
	printf("%d",f[1][0]);
	return 0;
}
posted @ 2017-11-05 16:51  CQzhangyu  阅读(342)  评论(0编辑  收藏  举报