洛谷2279

/*
树形DP 
 状态:
dp[i][0]:选自己
dp[i][1]:选了至少一个儿子
dp[i][2]:选了至少一个孙子
-----------------------------------这三种是覆盖了自己的
dp[i][3]: 儿子孙子全部覆盖
dp[i][4]:孙子全部覆盖
-----------------------------------这两种并没有覆盖自己
建议画一棵深度为3的完全二叉树直观观察状态   否则转移方程比较难懂

初始转移方程:
dp[i][0] = 1+Σmin(dp[j][0...4]);
要使选了根节点之后合法(整棵子树包括根节点被覆盖)必须使儿子的孙子全部覆盖 0~4状态满足 
dp[i][1] = min( dp[k][0] + Σ(j != k)min(dp[j][0...3]) );
要使选了一个儿子之后合法 由于儿子只可以覆盖到兄弟 所以孙子一定要全部被覆盖 即儿子的儿子一定覆盖  0~3满足 
dp[i][2] = min( dp[k][1] + Σ(j != k)min(dp[j][0...2]) );
使选了一个孙子之后合法  由于孙子最多只能覆盖到当前节点 所以儿子一定全部覆盖  即所有儿子本身要被覆盖  0~2满足 
dp[i][3] = Σdp[j][0...2];
要使儿子及孙子全部被覆盖  即儿子本身要被覆盖  0~2满足 
dp[i][4] = Σdp[j][0...3]; 
 要使孙子全部被覆盖  即儿子的儿子要全部被覆盖 0~3满足
::注意每种状态由儿子转移过来所以根的情况    要转化成对于儿子来说的情况
 
然后改进状态 因为每种转移方程至少有三种可能最后取其中较小的 故时间效率较低  令dp[i][k]表示min(dp[i][0],dp[i][1]....dp[i][k])且k>=2  因为上述转移方程最少都是0~2状态
 那么转移方程就大幅度化简了:
dp[i][0] = 1+Σdp[j][4];
直接由上面变形而来 
dp[i][1] = dp[i][4] + min(dp[k][0]-dp[k][3]);
选一个儿子  需保证所有孙子被覆盖 即 dp[i][4] 然后要选出一个儿子 将他从0~3状态变为选了自己(由于dp[i][4]中他是3状态所以要减去一个dp[k][3]) 取这个差值最小的儿子
dp[i][2] = dp[i][3] + min(dp[k][1]-dp[k][2]);
 选一个孙子 与上面类似  要保证所有儿子都被覆盖 即dp[i][3] 再将一个儿子从0~2状态变为0~1状态以保证覆盖他父节点
dp[i][3] = Σdp[j][2];
保证所有儿子被覆盖  儿子的0~2状态均符合条件 
dp[i][4] = Σdp[j][3];
保证所有儿子的儿子被覆盖 儿子的0~3状态均符合条件

别问我为什么dp[i][1]和dp[1][2]用到后面的状态  因为你只需要在过程中记下那一坨min的值 把3,4处理完后再算1,2
另外由于数据特殊性 编号大的节点一定是编号小的节点的后代  所以递推顺序直接到着推就好了
代码:(神犇的代码稍作改动) 
*/

#include<algorithm>
#include<cstdio>
using namespace std;
const int maxn=1000+10;
const int INF=(1<<30);
bool G[maxn][maxn];
int dp[maxn][5];
int main(){
    int n;scanf("%d",&n);
    for(int i=2,tmp;i<=n;i++){
          scanf("%d",&tmp);
		  G[tmp][i]=1;
    }
    for(int i=n;i>=1;i--){
        int x1=INF,x2=INF;
        dp[i][0]=1;
        for(int j=1;j<=n;j++)
        if(G[i][j]){
           dp[i][0]+=dp[j][4];
           dp[i][3]+=dp[j][2];
		   dp[i][4]+=dp[j][3];
           x1=min(x1,dp[j][0]-dp[j][3]);
           x2=min(x1,dp[j][1]-dp[j][2]);
        }
       dp[i][1]=dp[i][4]+x1;
       dp[i][2]=min(dp[i][3]+x2,min(dp[i][0],dp[i][1]));
	   dp[i][3]=min(dp[i][2],dp[i][3]);
       dp[i][4]=min(dp[i][3],dp[i][4]);
    }
    printf("%d",dp[1][2]);
    return 0;
}

  

posted @ 2018-04-18 07:02  lnyzo  阅读(86)  评论(0编辑  收藏  举报