BZOJ.1017.[JSOI2008]魔兽地图(树形DP 背包DP)
树形DP,考虑子节点对父节点的贡献。
设f[x][i][j]表示当前为x,用i个x去合成上一层装备,花费为j的最大价值。
由子节点转移时 是一个分组背包,需要一个辅助数组g[i][j]表示前i棵子树花费为j能贡献给x的最大价值。
那么 \(g[i][j] = max{g[i-1][j-k]+f[v][l*need[x]][k]}\)。\(need[x]\)为x需要子节点v的个数,\(l\)为合成x的个数,这个同样需要枚举。
那么对于每个\(l\),可以枚举用多少个x合成上一层,更新f,即 \(f[x][i][j] = max{g[all][j]+val[x]*(l-i)}\)。(\(g\)已经是要求合成\(l\)个x的价值)
处理完一棵子树,就可以简单地背包一下最大价值了。。
g[][]确实可以用一维,但是就不能进一步优化了,而且避免不了每次memset。(优化相当明显)
费用流显然做不了嘛。。
优化后:43108kb 1100ms
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
const int N=53,M=2002,INF=0x3f3f3f3f;
int n,m,Enum,H[N],nxt[N],to[N],need[N],dgr[N],val[N],cost[N],num[N],Ans[M],f[N][101][M],g[N][M];
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
inline void AddEdge(int u,int v,int w){
++dgr[v], to[++Enum]=v, nxt[Enum]=H[u], H[u]=Enum, need[Enum]=w;
}
void dfs(int x)
{
if(!H[x]){
num[x]=std::min(num[x],m/cost[x]);
for(int i=0; i<=num[x]; ++i)
for(int j=i; j<=num[x]; ++j)//限制不是 j*cost[x]<=m。。还有num[]限制。
f[x][i][j*cost[x]]=(j-i)*val[x];
return;
}
num[x]=100;
for(int i=H[x]; i; i=nxt[i])
dfs(to[i]), num[x]=std::min(num[x],num[to[i]]/need[i]), cost[x]+=need[i]*cost[to[i]];
num[x]=std::min(num[x],m/cost[x]);
memset(g,-0x3f,sizeof g);
g[0][0]=0;
for(int now,l=num[x]; ~l; --l)//当前要合成l个
{
now=0;
for(int i=H[x],v=to[i],w=l*need[i]; i; i=nxt[i],v=to[i],w=l*need[i],++now)
for(int j=0; j<=m; ++j)
if(g[now][j]>=0)//这个优化很有效
for(int k=0; k+j<=m; ++k)//g[0][..]不会被更新了,每次一定是由之前合法的转移来
g[now+1][j+k]=std::max(g[now+1][j+k],g[now][j]+f[v][w][k]);
for(int i=0; i<=l; ++i)
for(int j=0; j<=m; ++j)
f[x][i][j]=std::max(f[x][i][j],g[now][j]+(l-i)*val[x]);//不需要再枚举k用g[k]更新f[j-k],f只是由g[]转移来,且只能从当前g[]转移。
}
}
int main()
{
n=read(),m=read();
char s[3];
for(int x,v,i=1; i<=n; ++i)
{
val[i]=read(), scanf("%s",s);
if(s[0]=='A'){
x=read();
while(x--) v=read(),AddEdge(i,v,read());
}
else cost[i]=read(),num[i]=read();
}
memset(f,-0x3f,sizeof f);//会有非法状态。
for(int i=1; i<=n; ++i)
if(!dgr[i])
{
dfs(i);
for(int j=m; j; --j)
for(int k=1; k<=j; ++k)
Ans[j]=std::max(Ans[j],Ans[j-k]+f[i][0][k]);//根节点从f[i][0]转移就好了。
}
printf("%d",Ans[m]);
return 0;
}
优化前:42700kb 6968ms
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
const int N=53,M=2002,INF=0x3f3f3f3f;
int n,m,Enum,H[N],nxt[N],to[N],need[N],dgr[N],val[N],cost[N],num[N],Ans[M],f[N][101][M],g[M];
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
inline void AddEdge(int u,int v,int w){
++dgr[v], to[++Enum]=v, nxt[Enum]=H[u], H[u]=Enum, need[Enum]=w;
}
void dfs(int x)
{
if(!H[x]){
num[x]=std::min(num[x],m/cost[x]);
for(int i=0; i<=num[x]; ++i)
for(int j=i; j<=num[x]; ++j)//限制不是 j*cost[x]<=m。。还有num[]限制。
f[x][i][j*cost[x]]=(j-i)*val[x];
return;
}
num[x]=100;
for(int i=H[x]; i; i=nxt[i])
dfs(to[i]), num[x]=std::min(num[x],num[to[i]]/need[i]), cost[x]+=need[i]*cost[to[i]];
num[x]=std::min(num[x],m/cost[x]);
for(int l=num[x]; ~l; --l)//当前要合成l个
{//g[]不需要再清空了吧,数量递减g[]一定是递增的。当然这么写本来也不需要。
memset(g,-0x3f,sizeof g);//然而必须要清空。。之后的g[j]可能被之前本应非法-INF的g[j-k]给更新了。。
g[0]=0;//necessary
for(int i=H[x]; i; i=nxt[i])
for(int j=m; ~j; --j)//这要更新到0!(DP g[]的初始值)
{
int tmp=-INF;
for(int k=0; k<=j; ++k)
tmp=std::max(tmp,g[j-k]+f[to[i]][l*need[i]][k]);
g[j]=tmp;
}
for(int i=0; i<=l; ++i)
for(int j=0; j<=m; ++j)
f[x][i][j]=std::max(f[x][i][j],g[j]+(l-i)*val[x]);//不需要再枚举k用g[k]更新f[j-k],f只是由g[]转移来,且只能从当前g[]转移。
}
}
int main()
{
n=read(),m=read();
char s[3];
for(int x,v,i=1; i<=n; ++i)
{
val[i]=read(), scanf("%s",s);
if(s[0]=='A'){
x=read();
while(x--) v=read(),AddEdge(i,v,read());
}
else cost[i]=read(),num[i]=read();
}
memset(f,-0x3f,sizeof f);//会有非法状态。
for(int i=1; i<=n; ++i)
if(!dgr[i])
{
dfs(i);
for(int j=m; j; --j)
for(int k=1; k<=j; ++k)
Ans[j]=std::max(Ans[j],Ans[j-k]+f[i][0][k]);//根节点从f[i][0]转移就好了。
}
printf("%d",Ans[m]);
return 0;
}
BZOJ.1017.[JSOI2008]魔兽地图(树形DP 背包DP)的更多相关文章
- 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 ...
- [JSOI2008]魔兽地图(树形dp)
DotR (Defense of the Robots) Allstars是一个风靡全球的魔兽地图,他的规则简单与同样流行的地图DotA (Defense of the Ancients) Allst ...
- [luogu4037 JSOI2008] 魔兽地图 (树形dp)
传送门 Description DotR (Defense of the Robots) Allstars是一个风靡全球的魔兽地图,他的规则简单与同样流行的地图DotA (Defense of the ...
- [BZOJ1017][JSOI2008]魔兽地图DotR 树形dp
1017: [JSOI2008]魔兽地图DotR Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 2597 Solved: 1010[Submit][ ...
- 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]魔兽地图(动态规划)
[BZOJ1017][JSOI2008]魔兽地图(动态规划) 题面 BZOJ 洛谷 题解 状态设一下,\(f[i][j][k]\)表示第\(i\)个物品,有\(j\)个用于合成,总花费为\(k\)的最 ...
随机推荐
- [应用篇]第六篇 JSTL之自定义函数标签库
在之前的JSTL的总结中已经对函数标签库进行了一些说明,在这里我再一次重新整理一下! 自带函数标签库介绍 引入该标签库的方法为: <%@ taglib prefix="fn&quo ...
- 如何设置Ultraedit自动换行
有时候这会非常麻烦, 要让Ultraedit自动换行请按发下方法: 1. 点击菜单栏的"高级→配置",找到"编辑器→自动换行/制表符设置". 2. 然后,把&q ...
- Codeforces 923 D. Picking Strings
http://codeforces.com/contest/923/problem/D 题意: A-->BC , B-->AC , C-->AB , AAA-->empty 问 ...
- 第一、介绍Canvas
canvas能做什么? canvas是HTML5中的新元素,你可以使用javascript用它来绘制图形.图标.以及其它任何视觉性图像.它也可用于创建图片特效和动画.如果你掌握了完整的命令,你可以用c ...
- UVALive 6467
题目链接 : http://acm.sdibt.edu.cn/vjudge/contest/view.action?cid=2186#problem/C 题意:对于斐波那契数列,每个数都mod m , ...
- 图文详解 解决 MVC4 Code First 数据迁移
在使用Code first生成数据库后 当数据库发生更改时 运行程序就会出现数据已更改的问题 这时可以删除数据库重新生成解决 但是之前的数据就无法保留 为了保留之前的数据库数据 我们需要使用到C ...
- CSS marging相关
一.margin可以为负值 在盒模型中,内容区的width/height.padding.border都不能为负值,但是margin例外,它可以为负值. margin负值的本质,在于它改变了元素在普通 ...
- Java练习之使用StringBuilder
package string.demo; /* * 将数组变为字符串输出 */ public class StringBuilderTest { /** * @param args */ public ...
- MVC常用特性使用
简介 在以前的文章中,我和大家讨论如何用SingalR和数据库通知来完成一个消息监控应用. 在上一篇文章中,我介绍了如何在MVC中对MongoDB进行CRUD操作. 今天,我将继续介绍一些在开发中非常 ...
- React-Native 之 Navigator与NavigatorIOS使用
前言 学习本系列内容需要具备一定 HTML 开发基础,没有基础的朋友可以先转至 HTML快速入门(一) 学习 本人接触 React Native 时间并不是特别长,所以对其中的内容和性质了解可能会有所 ...