DotR (Defense of the Robots) Allstars是一个风靡全球的魔兽地图,他的规则简单与同样流行的地图DotA (Defense of the Ancients) Allstars。

DotR里面的英雄只有一个属性——力量。他们需要购买装备来提升自己的力量值,每件装备都可以使佩戴它的英雄的力量值提高固定的点数,所以英雄的力量值等于它购买的所有装备的力量值之和。装备分为基本装备和高级装备两种。基本装备可以直接从商店里面用金币购买,而高级装备需要用基本装备或者较低级的高级装备来合成,合成不需要附加的金币。装备的合成路线可以用一棵树来表示。

比如,Sange and Yasha的合成需要Sange,Yasha和Sange and Yasha Recipe Scroll三样物品。其中Sange又要用Ogre Axe, Belt of Giant Strength和 Sange Recipe Scroll合成。每件基本装备都有数量限制,这限制了你不能无限制地合成某些性价比很高的装备。

现在,英雄Spectre有M个金币,他想用这些钱购买装备使自己的力量值尽量高。你能帮帮他吗?他会教你魔法Haunt(幽灵附体)作为回报的。

Solution

极强的一道树形dp题。

我们设dp[i][j][k]表示当前正在做以i为根的子树,我们需要用j个物品用于向上合成,花费了k元能够获得的最大收益。

首先我们可以递归回溯求出每个点的价格和他的限制购买次数(这个东西是和儿子节点相关的)。

然后就枚举向上传多少个物品,在这里我们记录一个辅助数组g[i][j]表示做到了第i棵子树,背包体积为j时能够获得的最大收益。

但它可能有多个联通块,做完每个子树后我们再把它合并到一个数组里就可以了。

Code

#include<iostream>
#include<cstdio>
#include<cstring>
#define inf 1e9
#define N 55
#define M 2010
using namespace std;
int dp[N][N<<][M],g[N][M],head[N],w[N],va[N],l[N],f[N][M],tot,m,n,ji[N];
struct efr{
int n,to,l;
}an[];
inline void add(int u,int v,int l){
an[++tot].n=head[u];
an[tot].to=v;
an[tot].l=l;
head[u]=tot;
}
void dfs(int u){
if(!head[u]){
l[u]=min(l[u],m/va[u]);//这里求出这件物品的购买限制
for(int i=;i<=l[u];++i)
for(int j=i;j<=l[u];++j)
dp[u][i][j*va[u]]=(j-i)*w[u];//j 表示买了多少个
return;
}
l[u]=inf;
for(int i=head[u];i;i=an[i].n){
int v=an[i].to;
dfs(v);
l[u]=min(l[u],l[v]/an[i].l);//更新l得知
va[u]+=va[v]*an[i].l;//在这里递归计算u的价值
}
l[u]=min(l[u],m/va[u]);
memset(g,-0x3f,sizeof(g));
g[][]=;
for(int o=l[u];o>=;--o){//枚举上传几个
int cnt=;
for(int i=head[u];i;i=an[i].n){
int v=an[i].to;
cnt++;
for(int j=m;j>=;--j)
for(int k=;k<=j;++k)
g[cnt][j]=max(g[cnt][j],g[cnt-][j-k]+dp[v][o*an[i].l][k]);
}
for(int i=;i<=o;++i)
for(int j=;j<=m;++j)
dp[u][i][j]=max(dp[u][i][j],g[cnt][j]+w[u]*(o-i));
}
}
inline char rd(){
char c=getchar();
while(c!='A'&&c!='B')c=getchar();
return c;
}
int main(){
scanf("%d%d",&n,&m);
memset(dp,-0x3f,sizeof(dp));
char c;
for(int i=;i<=n;++i){
scanf("%d",&w[i]);
c=rd();
if(c=='B')scanf("%d%d",&va[i],&l[i]);
else{
int x,y,z;
scanf("%d",&x);
for(int j=;j<=x;++j){
scanf("%d%d",&y,&z);
add(i,y,z);
ji[y]=;
}
}
}
int cnt=;
for(int i=;i<=n;++i)
if(!ji[i]){
dfs(i);
cnt++;
for(int j=;j<=m;++j)
for(int k=;k<=j;++k)
f[cnt][j]=max(f[cnt][j],f[cnt-][j-k]+dp[i][][k]);
}
int ans=;
for(int i=;i<=m;++i)
ans=max(ans,f[cnt][i]);
cout<<ans;
return ;
}

[JSOI2008]魔兽地图(树形dp)的更多相关文章

  1. [luogu4037 JSOI2008] 魔兽地图 (树形dp)

    传送门 Description DotR (Defense of the Robots) Allstars是一个风靡全球的魔兽地图,他的规则简单与同样流行的地图DotA (Defense of the ...

  2. BZOJ.1017.[JSOI2008]魔兽地图(树形DP 背包DP)

    题目链接 树形DP,考虑子节点对父节点的贡献. 设f[x][i][j]表示当前为x,用i个x去合成上一层装备,花费为j的最大价值. 由子节点转移时 是一个分组背包,需要一个辅助数组g[i][j]表示前 ...

  3. bzoj1017 [JSOI2008]魔兽地图DotR——DP

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1017 好难想的状态啊!f[i][j][k]表示i号物品有j个向上贡献,一共花了k钱的最大力量 ...

  4. [BZOJ1017][JSOI2008]魔兽地图DotR 树形dp

    1017: [JSOI2008]魔兽地图DotR Time Limit: 30 Sec  Memory Limit: 162 MBSubmit: 2597  Solved: 1010[Submit][ ...

  5. BZOJ [JSOI2008]魔兽地图DotR

    1017: [JSOI2008]魔兽地图DotR Time Limit: 30 Sec  Memory Limit: 162 MBSubmit: 1243  Solved: 532[Submit][S ...

  6. 【bzoj1017】[JSOI2008]魔兽地图DotR

    1017: [JSOI2008]魔兽地图DotR Time Limit: 30 Sec  Memory Limit: 162 MBSubmit: 1658  Solved: 755[Submit][S ...

  7. 【BZOJ1017】[JSOI2008]魔兽地图(动态规划)

    [BZOJ1017][JSOI2008]魔兽地图(动态规划) 题面 BZOJ 洛谷 题解 状态设一下,\(f[i][j][k]\)表示第\(i\)个物品,有\(j\)个用于合成,总花费为\(k\)的最 ...

  8. BZOJ1017: [JSOI2008]魔兽地图DotR【树形DP】【玄学】

    Description DotR (Defense of the Robots) Allstars是一个风靡全球的魔兽地图,他的规则简单与同样流行的地图DotA (Defense of the Anc ...

  9. BZOJ1017 [JSOI2008]魔兽地图DotR 【树形dp + 背包dp】

    题目链接 BZOJ1017 题解 orz hzwer 树形dp神题 设\(f[i][j][k]\)表示\(i\)号物品恰好花费\(k\)金币,并将\(j\)个物品贡献给父亲的合成时的最大收益 计算\( ...

随机推荐

  1. PAT L2-020 功夫传人

    https://pintia.cn/problem-sets/994805046380707840/problems/994805059118809088 一门武功能否传承久远并被发扬光大,是要看缘分 ...

  2. 关于spring的源码的理解

    从最基础的Hello World开始. spring的Hello World就三行代码: public void test() { ApplicationContext context = new C ...

  3. [转帖]50个必知的Linux命令技巧,你都掌握了吗?

    50个必知的Linux命令技巧,你都掌握了吗? https://blog.51cto.com/lizhenliang/2131141 https://blog.51cto.com/lizhenlian ...

  4. [转帖]学习关于TTL

    自己简单试了一下在家里与在公司里面服务器的连接: C:\Users\Administrator>tracert oms.inspur.com 通过最多 个跃点跟踪 到 oms.inspur.co ...

  5. JavaMail入门第一篇 邮件简介及API概述

    现如今,电子邮件在我们的生活当中扮演着越来越重要的角色,我们每个人几乎都会与其打交道(至少时不时我们都会接收到莫名其妙的垃圾邮件),在工作中,使用邮件进行交流沟通,可以使我们的工作有迹可循,也显的较为 ...

  6. 剑指offer(12)

    来两道关于链表链接的题目: 题目一: 输入两个单调递增的链表,输出两个链表合成后的链表,当然我们需要合成后的链表满足单调不减规则. 本题要考虑到其中一条链表是空或者两个都是空的情况. 在每个链表安上一 ...

  7. python爬虫-1

    import resquests #import urllib.request from bs4 import BeautifulSoup from collections import Ordere ...

  8. Postman & API

    Postman & API https://www.getpostman.com/ https://www.getpostman.com/downloads/ Postman Canary h ...

  9. 996.ICU

    996.ICU https://github.com/996icu/996.ICU https://www.zhihu.com/question/317722302 LICENSE https://g ...

  10. 使用PHP对二维索引数组进行排序

    本例中 data 数组中的每个单元表示一个表中的一行.这是典型的数据库记录的数据集合. 例子中的数据如下: volume | edition -------+-------- 67 | 2 86 | ...