【動態規劃】通向自由的鑰匙

通向自由的钥匙(key.pas/c/cpp) 
   
通向自由的钥匙被放n个房间里,这n个房间由n-1 条走廊连接。但是每个房间里都有
特别的保护魔法,在它的作用下,我无法通过这个房间,也无法取得其中的钥匙。虽然我可
以通过消耗能量来破坏房间里的魔法,但是我的能量是有限的。那么,如果我最先站在 1
号房间(1 号房间的保护魔法依然是有效的,也就是,如果不耗费能量,我无法通过1号房
间,也无法取得房间中的钥匙) ,如果我拥有的能量为P,我最多能取得多少钥匙? 
 
输入数据 
  第一行包含两个非负整数,第一个为N,第二个为P。 
  接下来n 行 ,按1~n的顺序描述了每个房间。 第i+1行包含两个非负整数cost和 keys,
分别为第 i件房取消魔法需要耗费的能量和房间内钥匙的数量。 
  接下来 n-1行,每行两个非负整数 x,y,表示x 号房间和y 号是连通的。 
 
输出数据 
  一行一个整数,表示取得钥匙的最大值。 
 
样例 
输入:key.in 
5 5 
1 2 
1 1 
1 1 
2 3 
3 4 
1 2 
1 3 
2 4 
2 5 
 
输出: key.out 
7 
 
数据范围 
  对于 20%的测试数据,有n<=20 
  对于 30%的测试数据,有n<=30 
  对于所有测试数据,有p,n<=100, cost <= Maxint, keys<= Maxint 
一道較為基礎的樹形DP。
首先多叉轉二叉,然後再DP即可。
狀態:用f[i][j]表示i這顆子樹,使用j的能量能得到的最大鑰匙數。
轉移方程:f[i][j] = max(f[son[i]][k] + f[bro[i]][j - k - c[i]] + v[i], f[bro[i]][j])。
讀入邊的時候,首先讀成無向圖,再根據無向圖建樹,否則後果很嚴重;還有某個節點消耗的能量可以為0,所以j = 0並不能當作邊界條件。
ACCode:

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <bitset>

using std::max;

const char fi[] = "key.in";
const char fo[] = "key.out";
const int maxN = 110;
const int maxM = 110;
const int MAX = 0x3fffff00;
const int MIN = -MAX;

struct Edge {int dest; Edge *next; };
Edge *edge[maxN];
int f[maxN][maxM];
int bro[maxN];
int son[maxN];
int c[maxN];
int v[maxN];
int n, m;

  void init_file()
  {
    freopen(fi, "r", stdin);
    freopen(fo, "w", stdout);
  }
  
  void insert(int u, int v)
  {
    Edge *p = new Edge;
    p -> dest = v;
    p -> next = edge[u];
    edge[u] = p;
  }
  
  void readdata()
  {
    scanf("%d%d", &n, &m);
    for (int i = 1; i < n + 1; ++i)
      scanf("%d%d", c + i, v + i);
    for (int i = 1; i < n; ++i)
    {
      int u, v;
      scanf("%d%d", &u, &v);
      insert(u, v);
      insert(v, u);
    }
  }
  
  int DP(int i, int j)
  {
    if (i == 0) return 0;
    if (f[i][j]) return f[i][j];
    if (bro[i])
      f[i][j] = max(f[i][j], DP(bro[i], j));
    if (j >= c[i])
    for (int k = 0; k + c[i] < j + 1; ++k)
    {
      f[i][j] = max(f[i][j], DP(son[i], k)
        + DP(bro[i], j - k - c[i]) + v[i]);
    }
    return f[i][j];
  }
  
  void Build(int u, int Last)
  {
    for (Edge *p = edge[u]; p; p = p -> next)
    {
      int v = p -> dest;
      if (v != Last)
      {
        Build(v, u);
        bro[v] = son[u];
        son[u] = v;
      }
    }
  }

  void work()
  {
    Build(1, 0);
    DP(1, m);
    printf("%d", f[1][m]);
  }
  
int main()
{
  init_file();
  readdata();
  work();
  exit(0);
}


  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值