zoukankan      html  css  js  c++  java
  • 皇宫看守(树形dp)

    皇宫看守(树形dp)

     

     

     题解:

    不同于战略游戏那道题要求每边有人看守,即只能靠自己或者靠儿子,本题要求每个点有人看守,即对于点root可以靠自己靠儿子或靠父亲

    设dp[root][0/1/2]表示0靠自己1靠爸爸2靠儿子

    root靠自己可以从儿子的三种状态转移,但是要加上自己设看守的费用

    root靠爸爸可以让儿子靠自己或者靠儿子的儿子(好爸爸)

    注意这里!root靠儿子必须要有一个儿子靠自己,其余儿子靠自己或者靠儿子的儿子,实现见代码

    特别的,根结点不能靠爸爸,叶子结点不能靠儿子

    AC_Code:

     1 #include <bits/stdc++.h>
     2 using namespace std;
     3 typedef long long ll;
     4 const int maxn = 1500+10;
     5 const int mod = 1e9+7;
     6 const int inf = 0x3f3f3f3f;
     7 
     8 int cnt[maxn],dp[maxn][3], son[maxn][maxn], k[maxn],fa[maxn];
     9 int n,x;
    10 
    11 //dp[i][0]:靠自己可以看守到i所需的最小花费
    12 //dp[i][1]:靠爸爸可以看守到i所需的最小花费
    13 //dp[i][2]:靠儿子可以看守到i所需的最小花费
    14 
    15 void dfs(int root){
    16     if( !cnt[root] ){           //叶子节点
    17         dp[root][0] = k[root];
    18         dp[root][1] = 0;
    19         dp[root][2] = inf;
    20         return ;
    21     }
    22 
    23     for(int i=1;i<=cnt[root];i++){
    24         dfs( son[root][i] );
    25     }
    26     dp[root][0] = k[root];
    27     int f = 0;
    28     for(int i=1;i<=cnt[root];i++){
    29         dp[root][0] += min(dp[son[root][i]][1], min(dp[son[root][i]][0], dp[son[root][i]][2]));
    30         dp[root][1] += min(dp[son[root][i]][0], dp[son[root][i]][2]);
    31         dp[root][2] += min(dp[son[root][i]][0], dp[son[root][i]][2]);
    32 
    33         if( dp[son[root][i]][0]<=dp[son[root][i]][2] ){//标记有没有哪个儿子靠自己比靠儿子的花费少的
    34             f = 1;
    35         }
    36     }
    37     if( !f ){//若没有
    38         int minn = 0x3f3f3f3f;
    39         for(int i=1;i<=cnt[root];i++){
    40             minn = min(minn, dp[son[root][i]][0]-dp[son[root][i]][2]);
    41         }
    42         dp[root][2] += minn;
    43     }
    44 }
    45 
    46 int main()
    47 {
    48     memset(dp,0,sizeof(dp));
    49     scanf("%d",&n);
    50     for(int i=1;i<=n;i++){
    51         int x; scanf("%d",&x);
    52         scanf("%d%d",&k[x],&cnt[x]);
    53         for(int j=1;j<=cnt[x];j++){
    54             scanf("%d",&son[x][j]);
    55             fa[son[x][j]] = x;
    56         }
    57     }
    58     for(int i=1;i<=n;i++){
    59         if( !fa[i] ){
    60             dfs(i);
    61             int ans = min(dp[i][0], dp[i][2]);
    62             printf("%d
    ",ans);
    63             break;
    64         }
    65     }
    66     return 0;
    67 }

     参考博客:here

  • 相关阅读:
    临时记事本
    D版??班得瑞英文天籁CD13集(下载!!!)
    一个程序员的早上
    使用C#实现Morse码的输出
    面向对象编程的乐趣(TextBox.Text="")
    如何从MS Word的表格中提取指定单元格的数据
    使用Turbo C创建自己的软件包(即创建接口)
    使用C#读取Word表格数据
    一种光栅绘制直线的方法
    关于数据库设计的一个思索这样做是不好的
  • 原文地址:https://www.cnblogs.com/wsy107316/p/14076108.html
Copyright © 2011-2022 走看看