POJ 2342 (树形DP)

题目链接http://poj.org/problem?id=2342

题目大意:直属上司和下属出席聚会。下属的上司出现了,下属就不能参加,反之下属参加。注意上司只是指直属的上司。每个人出席的人都有一个快乐值,问最大的快乐值是多少。

解题思路

首先确定一下顶头上司是谁。

f[v]=u表示u是v的父亲,这样addedge完了之后,从1开始while(f[root]) root=f[root],最后root就是顶头上司,也就是dfs的起点了。

用dp[i][0]表示对于第i个人不出席的最大快乐值,用dp[i][1]表示出席的最大快乐值。

则dp[i][0]=dp[i][0]+max(dp[son][0],dp[son][1]) ,原因是上司不出席,下属可出席也可不出席,取个大的。

dp[i][1]=dp[i][1]+dp[son][0],原因是上司一旦出席,下属绝对不能出席。

其中dp[i][1]的初始值为这个人的快乐值。

则ans=max(dp[root][0],dp[root][1])

 

注意本题的dfs只是起到推进树结构作用,推完树结构之后,还是传统的DP转移方式,不是记忆化搜索。

 

#include "cstdio"

#include "vector"

#include "cstring"

using namespace std;

#define maxn 6005

int hap[maxn],f[maxn],dp[maxn][2],head[maxn],tol;

struct Edge

{

    int to,next;

}e[maxn];

void addedge(int u,int v)

{

    e[tol].to=v;

    e[tol].next=head[u];

    head[u]=tol++;

}

void dfs(int root)

{

    dp[root][1]=hap[root];

    for(int i=head[root];i!=-1;i=e[i].next) dfs(e[i].to);

    for(int i=head[root];i!=-1;i=e[i].next)

    {

        dp[root][0]+=max(dp[e[i].to][1],dp[e[i].to][0]);

        dp[root][1]+=dp[e[i].to][0];

    }

}

int main()

{

    //freopen("in.txt","r",stdin);

    int n,u,v;

    while(~scanf("%d",&n)&&n)

    {

        memset(f,0,sizeof(f));

        memset(dp,0,sizeof(dp));

        memset(head,-1,sizeof(head));

        for(int i=1; i<=n; i++) scanf("%d",&hap[i]);

        for(int i=1; i<=n; i++)

        {

            scanf("%d%d",&v,&u);

            if(u==0&&v==0) break;

            f[v]=u;

            addedge(u,v);

        }

        int root=1;tol=0;

        while(f[root]) root=f[root];

        dfs(root);

        int res=max(dp[root][0],dp[root][1]);

        printf("%d\n",res);

    }

}

 

13547096 neopenx 2342 Accepted 408K 0MS C++ 1182B 2014-10-20 12:18:43

你可能感兴趣的:(poj)