【BZOJ】【1017】【JSOI2008】魔兽地图Dotr
树形DP
一开始想:f[i][j]表示以 i 为根的子树,花 j 块钱能得到的最高力量值,结果发现转移的时候没法保证叶子结点的数量限制TAT
只好去膜拜题解了……在这里贴两篇泛型背包的文章吧:《背包九讲》、徐持衡《浅谈几类背包题》
vfk的酷炫姿势没看懂……这篇题解应该讲的是比较清楚的一篇>_> http://blog.csdn.net/baidu_20126217/article/details/40086029
这题算是把我对树形DP的闭塞理解给打通了一点。
我本认为树形DP只有用子节点的状态去更新父节点的状态,真是太天真了。
实际上这道题里是用子节点的状态合并得到父节点的状态。
首先,设出状态f[i][j][k]表示节点i对父亲的贡献为j付出的代价为k时i节点及其子树可以得到的最多能量。
dp当然要从初始状态推起咯。
那么对于那些叶子节点,也就是所谓的基本装备:
f[i][j][j*cost[i]]=(j-i)*power[i]
然而对于那些非叶子节点:
f[i][j][k]=max{g[k-r]+f[son][j*need[son]][r]};
这个方程具体点的解释可以理解为预算为k,拨给这个项目经费为r。
注意在这里我们并没有对f[i][j][k]中这一层中“私吞”部分进行统计,所以是j*need[son],也就是假设全部先上交。
此处g数组是对上一次f[i][j]的复制,防止出现值的误调用。
此处循环考虑的因素比较多,所以总不能一边调用f[i][j]一边更新f[i][j]吧。memcpy多方便。
然后我们开始统计私吞部分:
f[i][j][k]=max{f[i][j'][k]+(j'-j)*power[i]}
事实上,(j'-j)*power[i]就是没有用于合成(即上交)的i装备产生的能量。
非常巧妙,可惜这状态我想不到,还是经验问题。
Tips:注意挖掉一些非法状态和缩小lim范围。
/**************************************************************
Problem: 1017
User: Tunix
Language: C++
Result: Accepted
Time:8236 ms
Memory:48576 kb
****************************************************************/ //BZOJ 1017
#include<vector>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
using namespace std;
inline int getint(){
int v=,sign=; char ch=getchar();
while(ch<''||ch>''){ if (ch=='-') sign=-; ch=getchar();}
while(ch>=''&&ch<=''){ v=v*+ch-''; ch=getchar();}
return v*sign;
}
const int N=,INF=~0u>>;
typedef long long LL;
/******************tamplate*********************/
int n,m,ans=-INF;
struct node{int to,v;};
vector<node>G[N];
int f[N][][],g[],cost[N],num[N],str[N],fa[N];
void dfs(int x){
if (!G[x].size()){
num[x]=min(num[x],m/cost[x]);
F(i,,num[x])
F(j,i,num[x])
f[x][i][j*cost[x]]=(j-i)*str[x];
return;
}//DP边界:叶子结点
num[x]=INF;
rep(i,G[x].size()){
dfs(G[x][i].to);
// cost[x]+=cost[G[x][i].to]*G[x][i].v;
num[x]=min(num[x],num[G[x][i].to]/G[x][i].v);
}//预处理合成装备的num和cost
F(i,,num[x]) f[x][i][]=;
rep(i,G[x].size()){
int to=G[x][i].to;
F(j,,num[x]){
memcpy(g,f[x][j],sizeof f[x][j]);
memset(f[x][j],-,sizeof f[x][j]);
D(k,m,){
D(r,k,)
if (g[k-r]!=- && f[to][j*G[x][i].v][r]!=-)
f[x][j][k]=max(f[x][j][k],g[k-r]+f[to][j*G[x][i].v][r]);
ans=max(ans,f[x][j][k]);
}
}
}
F(i,,num[x]) F(j,i,num[x]) F(k,,m)
if (f[x][j][k]!=-)
f[x][i][k]=max(f[x][i][k],f[x][j][k]+(j-i)*str[x]),
ans=max(ans,f[x][i][k]);
}
int main(){
#ifndef ONLINE_JUDGE
freopen("1017.in","r",stdin);
freopen("1017.out","w",stdout);
#endif
n=getint(); m=getint();
F(i,,n){
fa[i]=i;
num[i]=INF;
cost[i]=;
}
char s1[];
F(i,,n){
str[i]=getint();
scanf("%s",s1);
if (s1[]=='B'){
cost[i]=getint();
num[i]=getint();
}else{
int c=getint();
F(j,,c){
int x=getint(),y=getint();
G[i].pb((node){x,y});
fa[x]=i;
}
}
}
int root=;
F(i,,n) if (fa[i]==i){root=i;break;}
memset(f,-,sizeof f);
dfs(root);
printf("%d\n",ans);
return ;
}
【BZOJ】【1017】【JSOI2008】魔兽地图Dotr的更多相关文章
- bzoj 1017: [JSOI2008]魔兽地图DotR【树形dp+背包】
bzoj上是一个森林啊--? dp还是太弱了 设f[i][j][k]为到点i,合成j个i并且花费k金币能获得的最大力量值,a[i]为数量上限,b[i]为价格,p[i]为装备力量值 其实这个状态设计出来 ...
- bzoj 1017 : [JSOI2008]魔兽地图DotR
比较难想的的一道树形dp. 看到这道题正常的思路应该是$f[i][j][k]$表示i这棵子树里买了j个i物品花费为k的最大收益. 但如果直接这么定义的话转移复杂度会很高,需要枚举j,枚举孩子,枚举k, ...
- 1017: [JSOI2008]魔兽地图DotR - BZOJ
Description DotR (Defense of the Robots) Allstars是一个风靡全球的魔兽地图,他的规则简单与同样流行的地图DotA (Defense of the Anc ...
- BZOJ.1017.[JSOI2008]魔兽地图(树形DP 背包DP)
题目链接 树形DP,考虑子节点对父节点的贡献. 设f[x][i][j]表示当前为x,用i个x去合成上一层装备,花费为j的最大价值. 由子节点转移时 是一个分组背包,需要一个辅助数组g[i][j]表示前 ...
- BZOJ [JSOI2008]魔兽地图DotR
1017: [JSOI2008]魔兽地图DotR Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 1243 Solved: 532[Submit][S ...
- 【bzoj1017】[JSOI2008]魔兽地图DotR
1017: [JSOI2008]魔兽地图DotR Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 1658 Solved: 755[Submit][S ...
- [BZOJ1017][JSOI2008]魔兽地图DotR 树形dp
1017: [JSOI2008]魔兽地图DotR Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 2597 Solved: 1010[Submit][ ...
- [bzoj1017][JSOI2008]魔兽地图 DotR (Tree DP)【有待优化】
Description DotR (Defense of the Robots) Allstars是一个风靡全球的魔兽地图,他的规则简单与同样流行的地图DotA (Defense of the Anc ...
- BZOJ1017: [JSOI2008]魔兽地图DotR【树形DP】【玄学】
Description DotR (Defense of the Robots) Allstars是一个风靡全球的魔兽地图,他的规则简单与同样流行的地图DotA (Defense of the Anc ...
- BZOJ1017: [JSOI2008]魔兽地图DotR
传送门 设$f[i][j][k]$表示对于第$i$个点,向父节点贡献$j$个已合成的装备,花费了$k$的代价,最多获得的力量值. 单纯的$f[i][j][k]$是很难转移的,主要原因是无法维护和其他儿 ...
随机推荐
- Knockout.Js官网学习(style绑定、attr绑定)
Style绑定 style绑定是添加或删除一个或多个DOM元素上的style值.比如当数字变成负数时高亮显示,或者根据数字显示对应宽度的Bar.(注:如果你不是应用style值而是应用CSS clas ...
- activity切换动画特效
效果图: 结构图: 测试代码: 布局: <?xml version="1.0" encoding="utf-8"?> <LinearLayou ...
- 一元三次方程 (codevs 1038)题解
[问题描述] 有形如:ax3+bx2+cx+d=0这样的一个一元三次方程.给出该方程中各项的系数(a,b,c,d均为实数),并约定该方程存在三个不同实根(根的范围在-100至100之间),且根与根之差 ...
- DevExpress后置代码中初始化SQL数据源的方法
//初始化SQL数据源的提供者和连接字符串 函数 OK public virtual void InitSqlDataSource_ConStr(SqlDataSource sql_ds) { Con ...
- Mysql 简单问题汇总(持续更新)
主从架构相关问题 问题现象:从机连接主机时报错 [ERROR] Slave I/O: error connecting to master 'repl@192.168.0.50:3306' - ret ...
- 10-排序5 PAT Judge
用了冒泡和插入排序 果然没有什么本质区别..都是运行超时 用库函数sort也超时 The ranklist of PAT is generated from the status list, whic ...
- Python脚本控制的WebDriver 常用操作 <九> 定位一组对象
下面将使用WebDriver来模拟操作定位一组对象的操作 测试用例场景 从上一节的例子中可以看出,webdriver可以很方便的使用find_element方法来定位某个特定的对象,不过有时候我们却需 ...
- EMVTag系列2《磁条等效数据》
Ø 57 磁条2等效数据 L: var. up to 19 -M(必备):此数据必须存在并提供给终端,终端在读应用数据过程中,如果没有读到必备数据,终端中止交易 按GB/T 17552,磁条2的数据 ...
- kettle报错 ../deploy does not exist, please create it.
具体错误如下: Xlib: extension "RANDR" missing on display "localhost:10.0". ::, INFO [K ...
- AppCan教你从零开始做开发
经常收到类似这样的提问:新手开发APP,要怎么学?我有满屏幕的文档和视频,然而并没有什么卵用,因为我不知道该从哪看起……今天的主要内容是教大家,如何在AppCan移动平台创建应用,引擎插件选择.证书管 ...