题目大意:给出一个树,每个节点有一个值,问从1开始走走k步最多能遍历到最大的值的和;
dp[i][j][k] 当i为零是表示如果从j点开始遍历以j为跟的子树,走k步后回到原点的最大值。
dp[i][j][k] 当i为零是表示如果从j点开始遍历以j为跟的子树,走k步后不回到原点的最大值。

我们知道的是如果能走k步最大值为maxa那么走j>k步至少为maxa,所以每次用子树的和当前节点做背包,0+0= 0,0+1=1,1+0=1;

由于避免重复我是用了一个类似于滚动数组的东西处理的背包部分,但是我后来发现那样好蠢还不好写还费时其实将一类中的某一个放入背包中完全能将背包倒过来然后每到一格将所有的东西装进去就行了。

第一次的代码是这样的

#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
const int maxa = ;
const int maxn = ;
int dp[][maxa][maxa];//0表示能回到当前位置,1表示不回到
int edge[maxa][maxa];
int vv[maxa];
int num[maxa];
int n, p;
int dp1[maxa];
//#define max(a,b) a>b?a:b
void dfs(int a, int fa){
for(int i = ; i < num[a]; i++){
int k = edge[a][i];
if(k != fa){
dfs(k, a);
}
}
for(int i = ; i <= p; i++)
dp[][a][i] = dp[][a][i] = vv[a];
for(int u = ; u < num[a]; u++){
int v = edge[a][u];
if(v != fa){
for(int i = ; i <= p; i++)
dp1[i] = vv[a];
for(int i = ; i <= p; i++){
for(int k = ; k <= p; k++){
if(i + k + <= p){
dp1[i+k+] = max(dp1[i+k+], dp[][a][i]+dp[][v][k]);
}
if(i+k+ <= p)
dp1[i+k+] = max(dp1[i+k+], dp[][a][i]+dp[][v][k]);
}
}
for(int i = ; i <= p; i++){
dp[][a][i] = max(dp[][a][i], dp1[i]);
}
for(int i = ; i <= p; i++)
dp1[i] = vv[a];
for(int i = ; i <= p; i++){
for(int k = ; k <= p; k++){
if(i+k+ <= p){
dp1[i+k+] = max(dp1[i+k+],dp[][a][i]+ dp[][v][k]);
}
}
}
for(int i = ; i <= p; i++){
dp[][a][i] = max(dp[][a][i],dp1[i]);
}
}
}
}
int main(){
while(scanf("%d%d", &n, &p)!=EOF){
int a, b;
memset(num, , sizeof(num));
for(int i = ; i <= n; i++){
scanf("%d", &vv[i]);
}
for(int i = ; i < n; i++){
scanf("%d%d", &a, &b);
edge[a][num[a]++] = b;
edge[b][num[b]++] = a;
}
//memset(dp, 0, sizeof(dp));
dfs(, );
printf("%d\n", dp[][][p]);
}
}
/*
8 6
1 1 1 1 1 1 1 4
1 2
1 3
1 4
1 5
1 6
1 7
1 8
*/

改后是这样的

#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
const int maxa = ;
const int maxn = ;
int dp[][maxa][maxa];//0表示能回到当前位置,1表示不回到
int edge[maxa][maxa];
int vv[maxa];
int num[maxa];
int n, p;
//#define max(a,b) a>b?a:b
void dfs(int a, int fa){
for(int i = ; i < num[a]; i++){
int k = edge[a][i];
if(k != fa){
dfs(k, a);
}
}
for(int i = ; i <= p; i++)
dp[][a][i] = dp[][a][i] = vv[a];
for(int u = ; u < num[a]; u++){
int v = edge[a][u];
if(v != fa){
for(int i = p; i >= ; i--){
for(int k = ; k <= i; k++){
if(i - k- >= ){
dp[][a][i] = max(dp[][a][i], dp[][a][k]+dp[][v][i-k-]);
dp[][a][i] = max(dp[][a][i], dp[][a][k]+dp[][v][i-k-]);
}if(i - k - >= ){
dp[][a][i] = max(dp[][a][i], dp[][a][k]+dp[][v][i-k-]);
}
}
}
}
}
}
int main(){
while(scanf("%d%d", &n, &p)!=EOF){
int a, b;
memset(num, , sizeof(num));
for(int i = ; i <= n; i++){
scanf("%d", &vv[i]);
}
for(int i = ; i < n; i++){
scanf("%d%d", &a, &b);
edge[a][num[a]++] = b;
edge[b][num[b]++] = a;
}
//memset(dp, 0, sizeof(dp));
dfs(, );
printf("%d\n", max(dp[][][p], dp[][][p]));
}
}
/*
8 6
1 1 1 1 1 1 1 4
1 2
1 3
1 4
1 5
1 6
1 7
1 8
*/

poj2486的更多相关文章

  1. poj2486 Apple Tree (树形dp+分组背包)

    题目链接:https://vjudge.net/problem/POJ-2486 题意:一棵点权树,起点在1,求最多经过m条边的最大点权和. 思路: 树形dp经典题.用3维状态,dp[u][j][0/ ...

  2. POJ2486 Apple Tree(树形DP)

    题目大概是一棵树,每个结点都有若干个苹果,求从结点1出发最多走k步最多能得到多少个苹果. 考虑到结点可以重复走,容易想到这么个状态: dp[u][k][0]表示在以结点u为根的子树中走k步且必须返回u ...

  3. POJ-2486 Apple Tree (树形DP)

    题目大意:一棵点带权有根树,根节点为1.从根节点出发,走k步,求能收集的最大权值和. 题目分析:从一个点向其某棵子树出发有三种可能的情况: 1.停留在那棵子树上: 2.再回到这个点: 3.经过这个点走 ...

  4. POJ2486 - Apple Tree(树形DP)

    题目大意 给定一棵n个结点的树,每个结点上有一定数量的苹果,你可以从结点1开始走k步(从某个结点走到相邻的结点算一步),经过的结点上的苹果都可以吃掉,问你最多能够吃到多少苹果? 题解 蛋疼的问题就是可 ...

  5. poj2486 Apple Tree (树形dp)

    题意:有一颗苹果树,树上的u节点上有num[u]个苹果,树根为1号节点,囧king从根开始走,没走到一个节点就把接点上的苹果吃光,问囧king在不超过k步的情况下最多吃多少个苹果. 解题思路:处理出两 ...

  6. poj2486 Apple Tree【区间dp】

    转载请注明出处,谢谢:http://www.cnblogs.com/KirisameMarisa/p/4374766.html   ---by 墨染之樱花 [题目链接]http://poj.org/p ...

  7. poj2486(树形dp)

    题目链接:http://poj.org/problem?id=2486 题意:一颗树,n个点(1-n),n-1条边,每个点上有一个权值,求从1出发,走m步,最多能遍历到的权值. 分析:非常不错的树形d ...

  8. POJ2486 Apple Tree

    Time Limit: 1000MS   Memory Limit: 65536KB   64bit IO Format: %lld & %llu Description Wshxzt is ...

  9. POJ2486 Apple Tree 【树上背包】

    一句话题意:一棵树,一共n个点,每个点上有一个权值,求从1出发,走k步,最多能遍历到的权值.可以往回走. 第一(二)道树上背包题,先是看了dalao的题解,改了一点就过样例了.然而....TLE??? ...

随机推荐

  1. MemCached高级缓存

    MemCached高级缓存配置 Memcache相关介绍: memcache 是一个高性能的分布式的内存对象缓存系统,它能够存储各种各样的的数据,包括图片,视频,文件等等.缓存功能. DB.数据源-- ...

  2. javascript 网页图标音乐切换

    图片名称 sprite.zip <!doctype html> <html> <head> </head> <style> .css{ po ...

  3. 如何修改WAMP中mysql默认空密码&重新登录phpmyadmin

    WAMP安装好后,mysql密码是为空的,那么要如何修改呢?其实很简单,通过几条指令就行了,下面我就一步步来操作. 首先,通过WAMP打开mysql控制台. 提示输入密码,因为现在是空,所以直接按回车 ...

  4. symfony框架在中国移动cmwap网络下访问的问题

    最近用symfony框架给手机app做后台,发现在中国移动cmwap网络下会出现问题,所有请求都路由到根路径了 左图为原始$_SERVER对象,右图为$request->server对象由于re ...

  5. Bootstrap 基础教程

    Bootstrap——来自Twitter的流行前端框架 一:Bootstrap简介 Boostrap是一个非常受欢迎的前端开发框架,该框架极大的提高前端团队的开发效率. Bootstrap对常见的CS ...

  6. 深入理解C#:编程技巧总结(二)

    原创文章,转载请注明出处! 以下总结参阅了:MSDN文档.<C#高级编程>.<C#本质论>.前辈们的博客等资料,如有不正确的地方,请帮忙及时指出!以免误导! 在上一篇 深入理解 ...

  7. MySQL添加中文字符集问题 --- java.sql.SQLException: Incorrect string value

    今天在做开关记录的时候,数据库表中description字段用于记录当前版本的描述(需要存储中文),在测试程序的时候发现如果用户输入中文,那么后台会报错.错误信息如下: java.sql.SQLExc ...

  8. 转:用 git 下载 uboot 源码

    1. 起因: 想下载 uboot 源码,原先的方法都是下载压缩包,然后放到虚拟机上的 Ubuntu ,再解压. 在看 uboot 源码的时候,发现 v2016.01 版本的uboot中关于 board ...

  9. 转:DDR3详解(以Micron MT41J128M8 1Gb DDR3 SDRAM为例)之一

    转载来自:http://blog.csdn.net/shanghaiqianlun/article/details/6976804 作者:shanghaiqianlun的专栏 1.结构框图: 2.管脚 ...

  10. ECharts使用心得——矩阵树图

    1. 缘起 最近实验室 boss 让用矩形树图做一下研究数据的展示,囧,矩形树图是什么鬼.问了问师兄,说用 R 语言可以实现,让我去研究一下. = =,那这周不是不用干别的了. 刚好最近在研究百度出的 ...