思路不算很难,但细节处理很麻烦

前面建图、多叉转二叉,以及确定dp处理序列的过程都是套路,dp的状态转移过程以注释的形式阐述

 #include <cstdio>
 #include <cstring>
 #include <algorithm>
 #include <queue>

 int N,M,K;

 struct Edge
 {
     int to,next;
     int weight;
     void assign(int t,int n,int w)
     {
         to=t; next=n; weight=w;
     }
 };

 Edge elist[];
 ];
 ];
 ][]; //0 is left and 1 is right
 ];
 int ecnt;

 void initE()
 {
     memset(head,-,sizeof(head));
     memset(vis,,sizeof(vis));
     memset(child,-,sizeof(child));
     weight[]=;
     ecnt=;
 }

 inline void addEdge(int from,int to,int weight)
 {
     elist[ecnt].assign(to,head[from],weight);
     head[from]=ecnt++;
     elist[ecnt].assign(from,head[to],weight);
     head[to]=ecnt++;
 }

 bool input()
 {
     scanf("%d%d%d",&N,&M,&K);
     bool ok=true;
     >N) ok=false; //输出-1的情况
     initE();
     int a,b,c;
     ;i<N;i++)
     {
         scanf("%d%d%d",&a,&b,&c);
         addEdge(a,b,c);
     }
     return ok;
 }

 void buildBinaryTree(int cur)
 {
     vis[cur]=true;
     ;
     ;e=elist[e].next)
     {
         int& to=elist[e].to;
         int& w=elist[e].weight;
         if(!vis[to])
         {
             weight[to]=w;
             )
                 child[cur][]=to;
             else
                 child[last][]=to;
             last=to;
             buildBinaryTree(to);
         }
     }
 }

 ][][];
 /*
 dp[n][m][k]:处理到第n个节点时,大头已经吃掉了m个果子,其父节点将被(k=1)或不被(k=0)大头吃,此情况下的最优值
 dp[n]的处理范围:孩子兄弟二叉树中,以节点n为根的子树
 dp的初值见下方代码,ans=dp[v][K-1][1],v为孩子兄弟二叉树中1号节点的左孩子
 */

 std::queue<int> seq;

 void getSeq(int cur)
 {
     ]!=-) getSeq(child[cur][]);
     ]!=-) getSeq(child[cur][]);
     seq.push(cur);
 } //先处理右孩子,再处理左孩子,最后处理自身

 /*
 M>=3时,小头的总难受值的最小值必为0
 取两个小头A和B,奇数层的果子由A吃,偶数层的果子由B吃,这样难受值必为0
 所以只需考虑大头的难受值
 */

 int solve_aux_3()
 //若无特殊说明:solve_aux函数注释中提到的有关树的概念均指孩子兄弟二叉树,opt代指当前决策的较优值,“不吃”和“吃”均指大头
 {
     int ans;
     while(!seq.empty())
     {
         int cur=seq.front();
         seq.pop();
         );
         ]!=-) st|=;
         ]!=-) st|=;
         ) //cur是叶子节点
         {
             dp[cur][][]=dp[cur][][]=;
             dp[cur][][]=;
             dp[cur][][]=weight[cur];
         }
         ) //只有右孩子,状态转移和线性dp类似
         {
             ];
             dp[cur][][]=;
             ;i<K;i++)
                 dp[cur][i][]=std::min(dp[rc][i][],dp[rc][i-][]); //对于当前果子,opt=min(不吃,吃)
             dp[cur][][]=;
             ;i<K;i++)
                 dp[cur][i][]=std::min(dp[rc][i][],dp[rc][i-][]+weight[cur]); //opt=min(不吃,吃)
         }
         ) //只有左孩子
         {
             ];
             ) ans=dp[lc][K-][]; //最终答案
             else
             {
                 dp[cur][][]=dp[cur][][]=;
                 ;i<K;i++)
                     dp[cur][i][]=std::min(dp[lc][i][],dp[lc][i-][]); //opt=min(不吃,吃)
                 ;i<K;i++)
                     dp[cur][i][]=std::min(dp[lc][i][],dp[lc][i-][]+weight[cur]); //opt=min(不吃,吃)
             }
         }
         else //st=3,既有左孩子又有右孩子,最复杂的情况
         {
             ];
             ];
             dp[cur][][]=dp[cur][][]=;
             ;i<K;i++) //dp[cur][i][0]
             {
                 ;j<=i;j++) //不吃当前的果子
                     dp[cur][i][]=std::min(dp[lc][j][]+dp[rc][i-j][],dp[cur][i][]); //分配i,取最优的分配方案
                 ;j<i;j++) //吃当前的果子
                     dp[cur][i][]=std::min(dp[lc][j][]+dp[rc][i-j-][],dp[cur][i][]);
             }
             ;i<K;i++) //dp[cur][i][1]
             {
                 ;j<=i;j++) //不吃
                     dp[cur][i][]=std::min(dp[cur][i][],dp[lc][j][]+dp[rc][i-j][]);
                 ;j<i;j++) //吃
                     dp[cur][i][]=std::min(dp[cur][i][],dp[lc][j][]+dp[rc][i-j-][]+weight[cur]);
             }
         }
     }
     return ans;
 }

 /*
 M=2时的dp方程与M>2时形式类似,但细节上有所不同(包括初值的设置和状态转移)
 此时大头和小头的难受值必须同时考虑
 */

 int solve_aux_2()
 {
     int ans;
     while(!seq.empty())
     {
         int cur=seq.front();
         seq.pop();
         );
         ]!=-) st|=;
         ]!=-) st|=;
         )
         {
             dp[cur][][]=dp[cur][][]=;
             dp[cur][][]=dp[cur][][]=weight[cur]; //注意这里赋初值的差异
             //dp[cur][0][0]表示原树中当前节点和父节点的果子都由小头吃,所以该段树枝的难受值也必须考虑
             //类似的差异会在下方用“!”标注,请读者自行体会
         }
         )
         {
             ];
             dp[cur][][]=dp[rc][][];
             dp[cur][][]=dp[rc][][]+weight[cur]; //!
             ;i<K;i++)
             {
                 dp[cur][i][]=std::min(dp[rc][i][]+weight[cur],dp[rc][i-][]); //!
                 dp[cur][i][]=std::min(dp[rc][i][],dp[rc][i-][]+weight[cur]);
             }
         }
         )
         {
             ];
             ) ans=dp[lc][K-][];
             else
             {
                 dp[cur][][]=dp[lc][][]; //!
                 dp[cur][][]=dp[lc][][]+weight[cur];
                 ;i<K;i++)
                 {
                     dp[cur][i][]=std::min(dp[lc][i][]+weight[cur],dp[lc][i-][]); //!
                     dp[cur][i][]=std::min(dp[lc][i][],dp[lc][i-][]+weight[cur]);
                 }
             }
         }
         else
         {
             ];
             ];
             dp[cur][][]=dp[lc][][]+dp[rc][][]; //!
             dp[cur][][]=dp[lc][][]+dp[rc][][]+weight[cur];
             ;i<K;i++)
             {
                 ;j<=i;j++)
                     dp[cur][i][]=std::min(dp[cur][i][],dp[lc][j][]+dp[rc][i-j][]+weight[cur]); //!
                 ;j<i;j++)
                     dp[cur][i][]=std::min(dp[cur][i][],dp[lc][j][]+dp[rc][i-j-][]);
                 ;j<=i;j++)
                     dp[cur][i][]=std::min(dp[cur][i][],dp[lc][j][]+dp[rc][i-j][]);
                 ;j<i;j++)
                     dp[cur][i][]=std::min(dp[cur][i][],dp[lc][j][]+dp[rc][i-j-][]+weight[cur]);
             }
         }
     }
     return ans;
 }

 int solve()
 {
     memset(dp,0x3f,sizeof(dp));
     buildBinaryTree();
     getSeq();
     ?solve_aux_2():solve_aux_3();
 }

 int main()
 {
     if(!input()) printf("-1");
     else printf("%d",solve());
     ;
 }

Vijos1523 NOI2002 贪吃的九头龙 树形dp的更多相关文章

  1. [NOI2002]贪吃的九头龙(树形dp)

    [NOI2002]贪吃的九头龙 题目背景 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是 说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的 ...

  2. [codevs1746][NOI2002]贪吃的九头龙

    [codevs1746][NOI2002]贪吃的九头龙 试题描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是说它出生的时候有九个头,而在成长的过程中,它有时 ...

  3. vojis1523 NOI2002 贪吃的九头龙

    描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落. 有一天, ...

  4. [NOI2002] 贪吃的九头龙

    题目类型:树形DP 传送门:>Here< 题意:有一只九头龙要吃了一颗树,给出一棵\(N\)个节点的带边权的树.九头龙有\(M\)个头,其中一个是大头,大头要吃恰好\(K\)个节点,其他头 ...

  5. 洛谷 P4362 [NOI2002]贪吃的九头龙

    https://www.luogu.org/problemnew/show/P4362 首先有个很显然的dp:ans[i][j][k]表示i节点用j号头,i节点为根的子树中共有k个点用大头时i节点为根 ...

  6. Vijos1523贪吃的九头龙【树形DP】

    贪吃的九头龙 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头 ...

  7. Vijos 1523 贪吃的九头龙 【树形DP】

    贪吃的九头龙 背景 安徽省芜湖市第二十七中学测试题 NOI 2002 贪吃的九头龙(dragon) Description:OfficialData:OfficialProgram:Converted ...

  8. 贪吃的九头龙(tyvj P1523)

    T2 .tyvj   P1523贪吃的九头龙 描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于 ...

  9. codevs1746 贪吃的九头龙

    [问题描述]传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落.有一 ...

随机推荐

  1. bzoj1014

    动态询问LCP,所以我们不好用后缀数组考虑使用维护序列问题的splay+hash求LCP这里mark一下,hash求LCP常用mo=9875321自然溢出的话交上去莫名其妙WA了这里树上某节点hash ...

  2. bzoj1706

    倍增好题,f[p,i,j]表示i到j经过了2^p条边走过的最短路径显然f[p+1]可以由f[p]转移来然后对n二进制拆分累加即可 ; ..,..] of int64; f,pf:..] of int6 ...

  3. bzoj1954 poj3764

    对于xor有一个非常重要的性质A xor B xor B=A 并且满足交换律和结合律这道题是求无根树上最长的xor路径我们知道,无根树的题目我们都是要想办法转化为有根树来处理当我们确定了一个根,根到每 ...

  4. BZOJ3439: Kpm的MC密码

    3439: Kpm的MC密码 Time Limit: 15 Sec  Memory Limit: 256 MBSubmit: 166  Solved: 79[Submit][Status] Descr ...

  5. 清除nginx静态资源缓存

    之前写过一篇如何配置nginx缓存及手动清除缓存的文章: http://www.cnblogs.com/Eivll0m/p/4921829.html 但如果有大量缓存需要清理,手动一条条清理就比较慢了 ...

  6. (转载)JavaScript中定义变量

    (转载)http://blog.163.com/xuxiaoqianhz@126/blog/static/165190577201061594421870/ JavaScript中定义变量有两种方式: ...

  7. JavaScript高级程序设计20.pdf

    用户代理检测 为了不在全局作用域中添加多余的变量,我们使用模块增强模式来封装检测脚本 以下是完整的用户代理字符串检测脚本,包括检测呈现引擎.平台.Window操作系统.移动设备和游戏系统 var cl ...

  8. HDU2037 今年暑假不AC 贪心算法

    贪心算法 : 贪心算法就是只考虑眼前最优解而忽略整体的算法, 它所做出的仅是在某种意义上的局部最优解, 然后通过迭代的方法相继求出整体最优解. 但是不是所有问题都可以得到整体最优解, 所以选择贪心策略 ...

  9. 图形变幻矩阵 Transforms

    https://developer.apple.com/library/mac/documentation/GraphicsImaging/Conceptual/drawingwithquartz2d ...

  10. TOYS - POJ 2318(计算几何,叉积判断)

    题目大意:给你一个矩形的左上角和右下角的坐标,然后这个矩形有 N 个隔板分割成 N+1 个区域,下面有 M 组坐标,求出来每个区域包含的坐标数.   分析:做的第一道计算几何题目....使用叉积判断方 ...