树dp:边覆盖,点覆盖

#493. 求树的最小支配集

问题描述

对于一棵n个结点的无根树,求它的最小支配集。 最小支配集:指从所有顶点中取尽量少的点组成一个集合,使得剩下的所有点都与取出来的点有边相连。顶点个数最小的支配集被称为最小支配集。

输入格式

第一行一个整数n,表示结点数。接下来n-1行,每行两个整数a,b,表示结点a和b有边。

输出格式

一行,一个整数,表示最小支配集中元素的个数。

输入样例

9
2 5
2 7
2 8
9 2
3 2
3 1
3 6
4 3

输出样例

2

限制与预定

1 < n <=100000

时间限制:1s

空间限制:128mb

#include<cstdio>
using namespace std;
const int maxn=1e5+5;
int n,f[maxn][3];
//0->儿子,1->父亲,2->自己 
inline int min(int a,int b){
    return(a<b?a:b);
}
inline void read(int &ans){
    ans=0;int b=1;
    char x=getchar();
    while(x<'0' || '9'<x){
    if(x=='-')b=-1;
    x=getchar();
    }
    while('0'<=x && x<='9'){
        ans=(ans<<3)+(ans<<1)+x-'0';
        x=getchar();
    }
    ans*=b;
}
int edge_count=0,to[maxn*2],next[maxn*2],first[maxn];
inline void add_edge(int x,int y){
    edge_count++;
    to[edge_count]=y;
    next[edge_count]=first[x];
    first[x]=edge_count;
}
void search(int root,int fa)
{
    int pos=0;
    for(int i=first[root];i;i=next[i]){
if(to[i]==fa)continue;

search(to[i],root);
f[root][2]+=f[ to[i] ][1];
f[root][1]+=min(f[ to[i] ][2],f[ to[i] ][0]);//+最小值

if( f[ to[i] ][2]-f[ to[i] ][0] < f[pos][2]-f[pos][0] )
pos=to[i];
}    
    f[root][2]++;
    if(f[root][1]>f[root][2])f[root][1]=f[root][2];
    bool fd=1;
    for( int i=first[root];i;i=next[i] ){
    if(to[i]==fa)continue;
    fd=0;
    if(to[i]==pos){
        f[root][0]+=f[pos][2];
        }
    else{
        f[root][0]+=min(f[to[i]][0],f[to[i]][2]);
        }
    }
f[root][0]+=fd;
//if(root==9||root==8||root==7||root==5)printf("%d %d",root,f[root][0]);
//if(f[root][0]||f[root][1]||f[root][2] )printf("%d",root);
//if(root==1)printf("%d",f[1][0]);
}
int main()
{
    //freopen("2.in","r",stdin);
    
    read(n);
    for( int i=1,a,b;i<n;i++){
        read(a);read(b);
        add_edge(a,b);
        add_edge(b,a);
    } 
    f[0][2]=0x7fffffff;
    search(1,0);
    printf("%d",min(f[1][0],min(f[1][1]+1,f[1][2])));
    return 0;
}

 

posted @ 2019-04-04 19:52  Tj1  阅读(295)  评论(0编辑  收藏  举报