以为很水的一道题,花了大半天的时间才搞定,比赛的时候卡在这题上了,伤不起啊。。。

题意:给一棵树,每个结点中有礼物,每个礼物有一个权值,某些结点中会有陷阱,你可以从任何一点出发,每个结点最多只能经过一次,最多掉进陷阱C次,求出可获得的礼物的最大值。

思路:典型的树形DP   ,状态可用dp[x][y][z]来表示,x代表以x为根结点的子树,y代表恰好经过了几个陷阱,z代表方向(0,1),表示从此子树进来或是出去。

dp[x][y][0]的含义是从子树x中出来,恰好经过y个陷阱所能获得的最大礼物值,dp[x][y][1]的含义是进入子树x,恰好经过y个陷阱所能获得的最大礼物值。

状态转移方程:

如果结点x有陷阱:dp[x][y][z]=max(dp[u][y-1][z])

如果结点x无陷阱:dp[x][y][z]=max(dp[u][y][z]),u是x的子结点

先把无根树转化为有根树,用一个全局变量记录答案,每求出一棵子树的dp值,就把答案更新一次:一种找出该子树中两棵不同的子子树,一颗出树,一颗入树,在两树通过父结点x连到一起,在该路径上的陷阱数不超过C的情况下,通过枚举两棵子子树上的陷阱数来更新答案,另一种是找出一棵子子树上的单条路径,在陷阱数不超过C的情况下,更新答案。

一定要注意该子树根结点有陷阱和无陷阱时要分开处理。

整体采用记忆化搜索实现。

细节太多,不再一一叙述,详情见代码:

#include <cstdio>
#include <vector>
#include <cstring>
#include <algorithm>
#define LL long long
using namespace std;
vector<int>v[50005];
int N,C,b[50005],r[50005];
LL a[50005],res,dp[50005][4][2];
bool vis[50005][4][2];
void root(int x)
{
for (int i=0; i<v[x].size(); ++i)
if (v[x][i]!=r[x])
{
r[v[x][i]]=x;
root(v[x][i]);
}
} LL dfs(int x,int y,int z); void update(int x)
{
vector<LL>ls[4],rs[4],e[4];
if (v[x].size()==1 && x)
{
if (!b[x]) res=max(res,a[x]);
else if (C) res=max(res,a[x]);
}
else if ((!x && v[x].size()==1) || (x && v[x].size()==2))
return;
else if (!b[x])
{
for (int j=0; j<4; ++j)
for (int i=0; i<v[x].size(); ++i)
if (v[x][i]!=r[x])
{
if (dfs(v[x][i],j,0))
{
ls[j].push_back(dfs(v[x][i],j,0)+a[x]);
rs[j].push_back(dfs(v[x][i],j,0)+a[x]);
}
else
{
ls[j].push_back(dfs(v[x][i],j,0));
rs[j].push_back(dfs(v[x][i],j,0));
}
if (dfs(v[x][i],j,1))
e[j].push_back(dfs(v[x][i],j,1)+a[x]);
else
e[j].push_back(dfs(v[x][i],j,1));
}
for (int j=0; j<4; ++j)
{
LL now=ls[j][0];
for (int i=1; i<ls[j].size(); ++i)
if (ls[j][i])
{
if (ls[j][i]>now) now=ls[j][i];
else ls[j][i]=now;
}
now=rs[j][rs[j].size()-1];
for (int i=ls[j].size()-2; i>=0; --i)
if (rs[j][i])
{
if (rs[j][i]>now) now=rs[j][i];
else rs[j][i]=now;
}
}
for (int i=0; i<=C; ++i)
for (int k=0; k<e[i].size(); ++k)
for (int j=0; j<=C-i; ++j)
if (e[i][k] && ((k && ls[j][k-1]) || (k<rs[j].size()-1 && rs[j][k+1])))
{
if (C && j==C) continue;
LL u=0;
if (k) u=max(u,ls[j][k-1]);
if (k<rs[j].size()-1) u=max(u,rs[j][k+1]);
res=max(res,u+e[i][k]-a[x]);
}
}
else
{
for (int j=0; j<4; ++j)
for (int i=0; i<v[x].size(); ++i)
if (v[x][i]!=r[x])
{
if (dfs(v[x][i],j,0))
{
ls[j].push_back(dfs(v[x][i],j,0)+a[x]);
rs[j].push_back(dfs(v[x][i],j,0)+a[x]);
}
else
{
ls[j].push_back(dfs(v[x][i],j,0));
rs[j].push_back(dfs(v[x][i],j,0));
}
if (dfs(v[x][i],j,1))
e[j].push_back(dfs(v[x][i],j,1)+a[x]);
else
e[j].push_back(dfs(v[x][i],j,1));
}
for (int j=0; j<4; ++j)
{
LL now=ls[j][0];
for (int i=1; i<ls[j].size(); ++i)
if (ls[j][i])
{
if (ls[j][i]>now) now=ls[j][i];
else ls[j][i]=now;
}
now=rs[j][rs[j].size()-1];
for (int i=ls[j].size()-2; i>=0; --i)
if (rs[j][i])
{
if (rs[j][i]>now) now=rs[j][i];
else rs[j][i]=now;
}
}
for (int i=0; i<C; ++i)
for (int k=0; k<e[i].size(); ++k)
for (int j=0; j<=C-i-1; ++j)
if (e[i][k] && ((k && ls[j][k-1]) || (k<rs[j].size()-1 && rs[j][k+1])))
{
if (j==C-1) continue;
LL u=0;
if (k) u=max(u,ls[j][k-1]);
if (k<rs[j].size()-1) u=max(u,rs[j][k+1]);
res=max(res,u+e[i][k]-a[x]);
}
}
} int main()
{
int T;
// freopen("1006.in","r",stdin);
// freopen("out.txt","w",stdout);
scanf("%d",&T);
while (T--)
{
int x,y;
scanf("%d%d",&N,&C);
for (int i=0; i<N; ++i) scanf("%I64d%d",&a[i],&b[i]);
for (int i=0; i<N; ++i) v[i].clear();
for (int i=1; i<N; ++i)
{
scanf("%d%d",&x,&y);
v[x].push_back(y);
v[y].push_back(x);
}
r[0]=-1;
root(0);
res=0;
memset(vis,0,sizeof(vis));
dfs(0,0,0);
/*for (int i=0; i<N; ++i)
for (int j=0; j<4; ++j)
for (int k=0; k<2; ++k)
printf("dp[%d][%d][%d]=%lld\n",i,j,k,dp[i][j][k]);*/
printf("%I64d\n",res);
}
return 0;
} LL dfs(int x,int y,int z)
{
if (vis[x][y][z]) return dp[x][y][z];
for (int i=0; i<4; ++i)
for (int j=0; j<2; ++j)
{
vis[x][i][j]=true;
dp[x][i][j]=0;
}
if (v[x].size()==1 && x)
{
if (!b[x])
dp[x][0][0]=dp[x][0][1]=a[x];
else
dp[x][1][0]=dp[x][1][1]=a[x];
}
else if (!b[x])
{
for (int i=0; i<v[x].size(); ++i)
if (v[x][i]!=r[x])
for (int j=0; j<4; ++j)
{
dp[x][j][0]=max(dp[x][j][0],dfs(v[x][i],j,0));
dp[x][j][1]=max(dp[x][j][1],dfs(v[x][i],j,1));
}
for (int i=0; i<4; ++i)
for (int j=0; j<2; ++j)
if (dp[x][i][j])
dp[x][i][j]+=a[x];
}
else
{
for (int i=0; i<v[x].size(); ++i)
if (v[x][i]!=r[x])
for (int j=1; j<4; ++j)
{
dp[x][j][0]=max(dp[x][j][0],dfs(v[x][i],j-1,0));
dp[x][j][1]=max(dp[x][j][1],dfs(v[x][i],j-1,1));
}
for (int i=1; i<4; ++i)
for (int j=0; j<2; ++j)
if (dp[x][i][j])
dp[x][i][j]+=a[x];
dp[x][1][1]=a[x];
if (!dp[x][1][0]) dp[x][1][0]=a[x];
}
update(x);
if (!b[x])
{
for (int i=0; i<C; ++i)
res=max(res,dp[x][i][0]);
for (int i=0;i<=C;++i)
res=max(res,dp[x][i][1]);
}
else
{
for (int i=1;i<=C;++i)
res=max(res,dp[x][i][0]);
for (int i=1;i<=C;++i)
res=max(res,dp[x][i][1]);
}
return dp[x][y][z];
}

  

hdu4616_Game_树形DP的更多相关文章

  1. poj3417 LCA + 树形dp

    Network Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 4478   Accepted: 1292 Descripti ...

  2. COGS 2532. [HZOI 2016]树之美 树形dp

    可以发现这道题的数据范围有些奇怪,为毛n辣么大,而k只有10 我们从树形dp的角度来考虑这个问题. 如果我们设f[x][k]表示与x距离为k的点的数量,那么我们可以O(1)回答一个询问 可是这样的话d ...

  3. 【BZOJ-4726】Sabota? 树形DP

    4726: [POI2017]Sabota? Time Limit: 20 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 128  Solved ...

  4. 树形DP+DFS序+树状数组 HDOJ 5293 Tree chain problem(树链问题)

    题目链接 题意: 有n个点的一棵树.其中树上有m条已知的链,每条链有一个权值.从中选出任意个不相交的链使得链的权值和最大. 思路: 树形DP.设dp[i]表示i的子树下的最优权值和,sum[i]表示不 ...

  5. 树形DP

    切题ing!!!!! HDU  2196 Anniversary party 经典树形DP,以前写的太搓了,终于学会简单写法了.... #include <iostream> #inclu ...

  6. BZOJ 2286 消耗战 (虚树+树形DP)

    给出一个n节点的无向树,每条边都有一个边权,给出m个询问,每个询问询问ki个点,问切掉一些边后使得这些顶点无法与顶点1连接.最少的边权和是多少.(n<=250000,sigma(ki)<= ...

  7. POJ2342 树形dp

    原题:http://poj.org/problem?id=2342 树形dp入门题. 我们让dp[i][0]表示第i个人不去,dp[i][1]表示第i个人去 ,根据题意我们可以很容易的得到如下递推公式 ...

  8. hdu1561 The more, The Better (树形dp+背包)

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=1561 思路:树形dp+01背包 //看注释可以懂 用vector建树更简单. 代码: #i ...

  9. bzoj2500: 幸福的道路(树形dp+单调队列)

    好题.. 先找出每个节点的树上最长路 由树形DP完成 节点x,设其最长路的子节点为y 对于y的最长路,有向上和向下两种情况: down:y向子节点的最长路g[y][0] up:x的次长路的g[x][1 ...

随机推荐

  1. dom4j解析xml获取所有的子节点并放入map中

    dom4j递归解析所有子节点 //解析返回的xml字符串,生成document对象 Document document = DocumentHelper.parseText(resultXml); / ...

  2. 关于javascript中的深拷贝问题

    一直在尝试为javascript找一个快捷可靠的对象深拷贝的方法,昨天突发奇想,把对象push到一个空数组里,然后对改数组通过concat()或slice()进行拷贝,然后取出数组的第一个元素复制给变 ...

  3. Leetcode 617 Merge Two Binary Trees 二叉树

    题意: 给定两棵树,将两棵树合并成一颗树 输入 Tree 1 Tree 2 1 2 / \ / \ 3 2 1 3 / \ \ 5 4 7 输出 合并的树 3 / \ 4 5 / \ \ 5 4 7 ...

  4. win10 uwp 使用 asp dotnet core 做图床服务器客户端

    原文 win10 uwp 使用 asp dotnet core 做图床服务器客户端 本文告诉大家如何在 UWP 做客户端和 asp dotnet core 做服务器端来做一个图床工具   服务器端 从 ...

  5. C#实现通过拼多多分享微信公众号实现查询优惠券、佣金比率

    主要实现功能:关注公众号的用户发送拼多多商品链接,后台程序通过链接查找商品优惠券或返佣情况. 说明:使用了niltor 封装的拼多多接口 github地址 ,但是需要注意可能会存在返回模型无法正确解析 ...

  6. Expression.Blend.4 Chapter 接口设计

    原文:Expression.Blend.4 Chapter 接口设计 发现网上关于silverlight,WPF美工系列的书籍一直很少,而且很多都是英文的.在公司无聊,开始进行翻译. 翻译的地方可能有 ...

  7. c#开发移动APP-Xamarin入门

    原文:c#开发移动APP-Xamarin入门 如果您在.net环境下做开发,并且对WPF技术有一定了解及应用,同时也想进入移动App开发领域,推荐使用Xamarin开发移动应用 关于Xamarin不做 ...

  8. Excel 2013永久取消超链接

    原文:Excel 2013永久取消超链接 在使用Excel的过程中,Excel会自动将网址转换为超链接,操作不当,容易误点,引起不必要的错误, 那么本篇博客就总结下如何在Excel 2013里永久取消 ...

  9. WPF中用于嵌入其他进程窗口的自定义控件(AppContainer)

    原文:WPF中用于嵌入其他进程窗口的自定义控件(AppContainer) 版权声明:本文为博主原创文章,转载请注明作者和出处 https://blog.csdn.net/ZZZWWWPPP11199 ...

  10. Angular语法(一)——展示数据

    双花括号{{}} 展示数据 title = 'Tour of Heroes'; myHero = 'Windstorm'; <h1>{{title}}</h1> <h2& ...