这题和之前那个HDU2616有着奇妙的异曲同工之处。。都是要求某个点能够到达的最大权重的地方。。。

但是,这题加了个限制,要求最多只能够踩到C个陷阱,一单无路可走或者命用光了,就地开始清算总共得分之和。

于是首先考虑,C的大小只有4,那么可以进行非常方便的状态转移,即将之前2616中的各个矩阵都加一维,设定为走到这一步的时候,可以踩得陷阱的个数——如果可以踩得陷阱的个数是0就意味着不能够行走,于是直接规定,所有是0条命的都自动为0。

之后按照上题的方式进行列举,有所不同的是,需要考虑下数组传递参数的方式,我的做法是将临时变量开到全局空间,这样就就可以保证不会爆栈什么的了。

考虑每个点,如果该电有陷阱,那么所有的转移,都必须按是少了一条命的结果,否则就是直接转移,同事我们认为,由于没有点数小于等于0的点,所以,必然可以得出,在同一个点上,当两个人的命数量不相等时,必然会有命多的“最大得分不小于命少的最大得分”。于是只需要普通的更新就是了。

#include<iostream>
#include<vector>
#include<string.h>
#include<stdio.h>
using namespace std; const long long MAXN = ;
vector<int>G[MAXN];
const long long LIMIT = ;
long long child[MAXN][LIMIT];
long long child_left[MAXN][LIMIT];
long long child_right[MAXN][LIMIT];
long long tmp[MAXN][LIMIT];
bool trap[MAXN];
long long arr[MAXN];
long long n, c; long long max(long long a, long long b)
{
return a > b ? a : b;
} void get_child(int now, int last)
{
int len = G[now].size();
for (int i = ; i<LIMIT; ++i)child[now][i] = ;
for (int i = ; i<len; ++i)
{
int tar = G[now][i];
if (tar == last)continue; get_child(tar, now);
if (trap[now])for (int i = ; i<LIMIT; ++i)child[now][i] = max(child[now][i], child[tar][i - ]);
else for (int i = ; i<LIMIT; ++i)child[now][i] = max(child[now][i], child[tar][i]);
}
for (int i = ; i<LIMIT; ++i)child[now][i] += arr[now];
}
void get_left(int now, int last)
{
int len = G[now].size();
if (last == -)
{
memset(child_left[now], , sizeof(child_left));
memset(tmp[now], , sizeof(tmp[now]));
}
else
{
if (trap[now]) for (int i = ; i<LIMIT; ++i) child_left[now][i] = tmp[now][i] = tmp[last][i - ];
else for (int i = ; i<LIMIT; ++i)child_left[now][i] = tmp[now][i] = tmp[last][i]; } for (int i = ; i<LIMIT; ++i)child_left[now][i] += arr[now]; long long ttmp[LIMIT]; memset(ttmp, , sizeof(ttmp));
for (int i = ; i<len; ++i)
{
int tar = G[now][i];
if (tar == last)continue;
for (int j = ; j<LIMIT; ++j)tmp[now][j] = max(tmp[now][j], ttmp[j]);
for (int j = ; j<LIMIT; ++j)tmp[now][j] += arr[now];
get_left(tar, now);
for (int j = ; j<LIMIT; ++j)tmp[now][j] -= arr[now];
if (trap[now]) for (int j = ; j<LIMIT; ++j)ttmp[j] = max(ttmp[j], child[tar][j - ]);
else for (int j = ; j<LIMIT; ++j) ttmp[j] = max(ttmp[j], child[tar][j]);
} }
void get_right(int now, int last)
{
int len = G[now].size(); if (last == -)
{
memset(child_right[now], , sizeof(child_right));
memset(tmp[now], , sizeof(tmp[now]));
}
else
{
if (trap[now]) for (int i = ; i<LIMIT; ++i) child_right[now][i] = tmp[now][i] = tmp[last][i - ];
else for (int i = ; i<LIMIT; ++i)child_right[now][i] = tmp[now][i] = tmp[last][i]; }for (int i = ; i<LIMIT; ++i)child_right[now][i] += arr[now]; long long ttmp[LIMIT]; memset(ttmp, , sizeof(ttmp));
for (int i = len - ; i >= ; --i)
{
int tar = G[now][i];
if (tar == last)continue;
for (int j = ; j<LIMIT; ++j)tmp[now][j] = max(tmp[now][j], ttmp[j]);
for (int j = ; j<LIMIT; ++j)tmp[now][j] += arr[now];
get_right(tar, now);
for (int j = ; j<LIMIT; ++j)tmp[now][j] -= arr[now];
if (trap[now]) for (int j = ; j<LIMIT; ++j)ttmp[j] = max(ttmp[j], child[tar][j - ]);
else for (int j = ; j<LIMIT; ++j) ttmp[j] = max(ttmp[j], child[tar][j]);
} } void init()
{
cin >> n >> c;
for (int i = ; i <= n; ++i)G[i].clear();
for (int i = ; i<n; ++i) cin >> arr[i] >> trap[i];
for (int i = ; i<n; ++i)
{
int a, b; cin >> a >> b;
G[a].push_back(b);
G[b].push_back(a);
}get_child(, -);
get_left(, -);
get_right(, -);
long long ans = ;
for (int i = ; i<n; ++i)
{
ans = max(ans, max(max(child_left[i][c], child_right[i][c]), child[i][c]));
}
cout << ans << endl;
}
int main()
{
cin.sync_with_stdio(false);
int tt; cin >> tt;
while (tt--)init(); return ;
}

HDU4616 树形DP+三次深搜的更多相关文章

  1. 【BZOJ-1060】时态同步 树形DP (DFS爆搜)

    1060: [ZJOI2007]时态同步 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2101  Solved: 595[Submit][Statu ...

  2. 树形dp——三色二叉树

    题目描述 一棵二叉树可以按照如下规则表示成一个由0.1.2组成的字符序列,我们称之为"二叉树序列S": 0 该树没有子节点 1S1 该树有一个子节点,S1为其二叉树序列 1S1S2 ...

  3. 深搜+DP剪枝 codevs 1047 邮票面值设计

    codevs 1047 邮票面值设计 1999年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题目描述 Description ...

  4. P2668 斗地主 dp+深搜版

    题目描述 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗地主中,牌的大小关系根据牌的数码表示如下:3<4< ...

  5. 方格取数(多线程dp,深搜)

    https://www.luogu.org/problem/P1004 题目描述 设有N×N的方格图(N≤9),我们将其中的某些方格中填入正整数,而其他的方格中则放入数字0.如下图所示(见样例): 某 ...

  6. 【BZOJ-1864】三色二叉树 树形DP

    1864: [Zjoi2006]三色二叉树 Time Limit: 1 Sec  Memory Limit: 64 MBSubmit: 659  Solved: 469[Submit][Status] ...

  7. 【BZOJ1864】[Zjoi2006]三色二叉树 树形DP

    1864: [Zjoi2006]三色二叉树 Description Input 仅有一行,不超过500000个字符,表示一个二叉树序列. Output 输出文件也只有一行,包含两个数,依次表示最多和最 ...

  8. BZOJ_1864_[Zjoi2006]三色二叉树_树形DP

    BZOJ_1864_[Zjoi2006]三色二叉树_树形DP 题意: 分析:递归建树,然后DP,从子节点转移. 注意到红色和蓝色没有区别,因为我们可以将红蓝互换而方案是相同的.这样的话我们只需要知道当 ...

  9. 洛谷 P2585 [ ZJOI 2006 ] 三色二叉树 —— 树形DP

    题目:https://www.luogu.org/problemnew/show/P2585 首先,三色其实记录两种状态:是绿色,不是绿色 即可,因为红蓝可以随意取反: 一开始因为懒得还原出树,所以写 ...

随机推荐

  1. 排序算法对比,步骤,改进,java代码实现

    前言 发现是时候总结一番算法,基本类型的增删改查的性能对比,集合的串并性能的特性,死记太傻了,所以还是写在代码里,NO BB,SHOW ME THE CODE! github地址:https://gi ...

  2. 创建有输出参数的存储过程并在c#中实现DataGridView分页功能

    不足之处,欢迎指正! 创建有输出参数的存储过程 if exists(select * from sysobjects where name='usp_getPage1') drop procedure ...

  3. Mongodb简介及基本操作

    一.简介 MongoDB是一款强大.灵活.且易于扩展的通用型数据库 MongoDB 是由C++语言编写的,是一个基于分布式文件存储的开源数据库系统. 在高负载的情况下,添加更多的节点,可以保证服务器性 ...

  4. mysql数据库初步了解

    一丶数据库服务器丶数据管理系统丶数据库丶表与记录的关系 记录:1 xxxx 3245646546(多个字段的信息组成一条记录,即文件中的一行内容) 表: Student.school,class_li ...

  5. RxJava 1升级到RxJava 2过程中踩过的一些“坑”

    RxJava2介绍 RxJava2 发布已经有一段时间了,是对 RxJava 的一次重大的升级,由于我的一个库cv4j使用了 RxJava2 来尝鲜,但是 RxJava2 跟 RxJava1 是不能同 ...

  6. 【java】使用URL和CookieManager爬取页面的验证码和cookie并保存

    使用java的net包和io包下的几个工具爬取页面的验证码图片并保存到本地. 然后可以把获取的cookie保存下来,做进一步处理.比如通过识别验证码,进一步使用验证码和用户名,密码,保存下来的cook ...

  7. cms-帖子幻灯图片上传

    package com.open1111.controller.admin; import java.io.File;import java.util.Date;import java.util.Ha ...

  8. 基于spark Mllib(ML)聚类实战

        写在前面的话:由于spark2.0.0之后ML中才包括LDA,GaussianMixture 模型,这里k-means用的是ML模块做测试,LDA,GaussianMixture 则用的是ML ...

  9. 利用photoshop制作gif图片

    首先准备你需要的几张素材图片 1.将素材图片根据发生的顺序放置在不同的图层 2.打开窗口下的时间轴 选择帧动画 3.创建第一张帧动画 选项卡右边这个按钮,点击这个选择新建帧 第一张图片显示其他的隐藏 ...

  10. ABAP和Hybris的源代码生成工具

    ABAP 有两种方式,一种是ABAP Code Composer, 细节可以查看我的博客Step by Step to generate ABAP code automatically using C ...