也是ural1039
描述

huyichen世子事件后,xuzhenyi成了皇上特聘的御前一品侍卫。

皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状;某些宫殿间可以互相望见。大内保卫森严,三步一岗,五步一哨,每个宫殿都要有人全天候看守,在不同的宫殿安排看守所需的费用不同。

可是xuzhenyi手上的经费不足,无论如何也没法在每个宫殿都安置留守侍卫。

帮助xuzhenyi布置侍卫,在看守全部宫殿的前提下,使得花费的经费最少。

格式

输入格式

输入文件中数据表示一棵树,描述如下:

第1行 n,表示树中结点的数目。

第2行至第n+1n+1行,每行描述每个宫殿结点信息,依次为:该宫殿结点标号i(0<in0<i≤n),在该宫殿安置侍卫所需的经费k,该点的儿子数m,接下来m个数,分别是这个节点的m个儿子的标号r1,r2,,rmr1,r2,⋯,rm

对于一个n(0<n15000<n≤1500)个结点的树,结点标号在1到n之间,且标号不重复。保证经费总和不超过2311231−1

输出格式

输出文件仅包含一个数,为所求的最少的经费。

样例1

样例输入1[复制]

 
6
1 30 3 2 3 4
2 16 2 5 6
3 5 0
4 4 0
5 11 0
6 5 0

样例输出1[复制]

 
25

限制

 

提示

选择3,4,2费用最小为25

来源

huyichen


 

题解

有一棵树,每个点或者相邻的点必须要驻扎人,求驻扎总费用最小,典型的树形动规

分析

对于节点i,有三种状态:1、自守   2、子守  3、父守

方程:

f[i,1]:=Σ(min{f[son,1],f[son,2],f[son,3]})+a[i]

f[i,2]:=Σ(min{f[son,1],f[son,2]})+m    //m的意思是所有son的自守与子守的差最小值,如果大于0就要加上这说明没有一个儿子是自守,不符合定义,就必须加上m,算作这个儿子守

f[i,3]:=Σ(f[son,2])          //son的自守情况已经被上面的情况包含了

第2种情况要特别注意,要求它的子结点中必须有一个是1状况的,所以令m=min{f[son[j],1]-f[son[j],2]},如果m>0说明在决策的时候,它的子结点没有一个是1状况的,这样就要加上m,否则令m=0.

这个方程看了一下午久才懂

AC代码

#include<iostream>
#include<cstdio>
#define MAX 1000000000
using namespace std;
int root,n,tot=0,num,c,m,kk;
int cost[1505];
int head[1505];
int from[1504];
int to[1504];
int f[1505][4];// 1自守,2子守,3父守
void add(int a,int b)
{
    tot++;
    to[tot]=b;
    from[tot]=head[a];
    head[a]=tot;
}
int _min(int a,int b)
{
    return a<b?a:b;
}
void treedp(int x)
{
    if(head[x]==0)//叶子节点
    {
        f[x][1]=f[x][2]=cost[x];
        return;
    }
    int m,t;
    f[x][1]=cost[x];
    f[x][2]=f[x][3]=0;
    m=MAX;
    while(head[x]>0)
    {
        t=to[head[x]];
        treedp(t);
        f[x][1]=f[x][1]+_min(f[t][1],_min(f[t][2],f[t][3]));
        f[x][2]=f[x][2]+_min(f[t][2],f[t][1]);
        f[x][3]=f[x][3]+f[t][2];
        if(m>f[t][1]-f[t][2])
            m=f[t][1]-f[t][2];
        head[x]=from[head[x]];
    }
    if(m>0) f[x][2]+=m;
}
int main()
{
    scanf("%d",&n);
    root=n*(n+1)/2;
    for(int i=1;i<=n;i++)
    {
        scanf("%d%d%d",&num,&c,&m);
        cost[num]=c;
        for(int j=1;j<=m;j++)
        {
            scanf("%d",&kk);
            root-=kk;
            add(num,kk);
        }
    }
    treedp(root);
    cout<<_min(f[root][1],f[root][2]);
    return 0;
}
View Code

 

posted on 2016-07-16 16:30  李万  阅读(480)  评论(0编辑  收藏  举报