poj 1463

题意:一城堡的所有的道路形成一个n个节点的树,如果在一个节点上放上一个士兵,那么和这个节点相连的边就会被看守住,

问把所有边看守住最少需要放多少士兵。

点覆盖问题

树形dp或二分图都可以过。

代码:

ContractedBlock.gif ExpandedBlockStart.gif View Code
 
   
#include < iostream >
#include
< fstream >
#include
< vector >

using namespace std;


typedef
struct e{
int data;
vector
< int > a;
}e;

e edge[
1501 ];
int dp[ 1501 ][ 2 ];
int v[ 1501 ][ 2 ];
int n;
int solve( int );

int solve1( int s){
int i,j,k;
if (v[s][ 1 ]) return dp[s][ 1 ];
v[s][
1 ] = 1 ;
dp[s][
1 ] = 1 ;
k
= edge[s].a.size();
for (i = 0 ;i < k;i ++ )
{
j
= edge[s].a[i];
dp[s][
1 ] += solve(j);
}
return dp[s][ 1 ];
}

int solve( int s){
int i,j,k,t = 0 ;
if (v[s][ 0 ]) return dp[s][ 0 ];
v[s][
0 ] = 1 ;
k
= edge[s].a.size();
for (i = 0 ;i < k;i ++ )
{
j
= edge[s].a[i];
t
+= solve1(j);
}
dp[s][
0 ] = min(solve1(s),t);
return dp[s][ 0 ];
}

void read(){
// ifstream cin("in.txt");
int i,j,k,s,t;
char c;
while (scanf( " %d " , & n) != EOF)
{
for (i = 0 ;i <= n;i ++ )
{
edge[i].data
= i;
edge[i].a.clear();
}
int start;
for (i = 1 ;i <= n;i ++ )
{
// cin>>j;
// cin>>c;
// cin>>c;
// cin>>k;
// cin>>c;
scanf( " %d:(%d) " , & j, & k);
if (i == 1 ) start = j;
for (s = 1 ;s <= k;s ++ )
{
// cin>>t;
scanf( " %d " , & t);

edge[j].a.push_back(t);
// edge[t].a.push_back(j);

}
}
memset(v,
0 , sizeof (v));
cout
<< solve(start) << endl;
}
}

int main(){
read();
return 0 ;
}

转载于:https://www.cnblogs.com/zhaozhe/archive/2011/02/24/1964117.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值