Steal 偷天换日 题解(From luoguBlog)
树形+背包 奇奇怪怪的dp。
考试的时候费了半天劲把题读完后思路基本正解,
然而也不知道为什么脑子鬼畜了一下打了个非递归建树?
而且链式前向星建边?
岔路口和藏品都搞成节点?
自己给自己找麻烦Orz。
于是输入爆炸,无法调试。
比较简便的打法是把权值左右儿子几个藏品各多少钱都塞到结构体里,一边建树一遍跑dp(在每个馆里跑背包然后把儿子的价值树形dp到根);
然而还是有很多坑点
所以恐怖如斯如机房里的各路大神仍然没有超过40分的:
边权*2因为要逃跑;
时限-1因为要逃跑;
玄妙的数组大小;
Return见祖宗 ;
#include<cstdio>
#include<iostream>
using namespace std;
typedef long long ll;
int tl;
int f[1010][1010];
struct item
{
int dis;
int bny;
int lc,rc;
int vl[45],fee[45];
}e[610];
void build(int r)
{
scanf("%d%d",&e[r].dis,&e[r].bny);
e[r].dis*=2;
if(!e[r].bny)
{
e[r].lc=r*2;build(r*2);
e[r].rc=r*2+1;build(r*2+1);
for(int i=e[r].dis;i<=tl;i++)
for(int j=0;j<=i-e[r].dis;j++)f[r][i]=max(f[r][i],f[e[r].lc][j]+f[e[r].rc][i-j-e[r].dis]);
}
else
{
for(int i=1;i<=e[r].bny;i++)
scanf("%d%d",&e[r].vl[i],&e[r].fee[i]);
for(int i=1;i<=e[r].bny;i++)
for(int j=tl;j>=e[r].fee[i];j--)
if(j-e[r].fee[i]>=e[r].dis)
f[r][j]=max(f[r][j],f[r][j-e[r].fee[i]]+e[r].vl[i]);
return ;
}
}
int main()
{
scanf("%d",&tl);
tl--;
build(1);
cout<<f[1][tl]<<endl;
return 0;
}
兴许青竹早凋,碧梧已僵,人事本难防。