题目大意:给出一个树,每个节点有一个值,问从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. redis配置文件redis.conf详细说明

    # By default Redis does not run as a daemon. Use 'yes' if you need it.# Note that Redis will write a ...

  2. 微星b85(b85i b85-gaming) 系列dsdt

    从网友得了一个b85-gaming 的dsdt,发现跟我的b85i的dsdt错误都是一样的. 发布上来给需要的人参考. 微星这个系列的dsdt不能用dsdt editor的fix功能,不然文件会越来越 ...

  3. NT头 IMAGE_NT_HEADER

    typedef struct_IMAGE_NT_HEADERS{ DWORD Signature; // 固定为 0x00004550 “PE00" IMAGE_FILE_HEADER Fi ...

  4. 使用john破解ubuntu(linux)9.10密码

    Title:使用john破解ubuntu(linux)9.10密码 --2011-11-23 15:00 ubuntu 9.10的账户密码加密方式改用sha512了,默认的john是破不了的,还好官方 ...

  5. assets文件夹资源的访问

    访问assets文件夹中的文件,分为以下几个步骤:1.在Activity里面调用getAssets(),获取AssetManager引用2.调用AssetManager.open(String fil ...

  6. 包含无数好东西的ownCloud

    ownCloud 是一个来自 KDE 社区开发的免费软件,提供私人的 Web 服务.当前主要功能包括文件管理(内建文件分享).音乐.日历.联系人等等,可在PC和服务器上运行. 简单来说就是一个基于Ph ...

  7. 为何某些公司不允许使用C++ STL?

    说几个STL的缺点吧,虽然都是在比较极端的情况下出现,但是对于一些大项目还是会遇到的 1. 代码膨胀问题每一个实例化过的模板类,都会膨胀出一份独立的代码,比如std::vector<std::s ...

  8. 关于标准C语言的预定义宏

    标准C语言预处理要求定义某些对象宏,每个预定义宏的名称一两个下划线字符开头和结尾,这些预定义宏不能被取消定义(#undef)或由编程人员重新定义.下面预定义宏表,被我抄了下来.__LINE__  当前 ...

  9. iOS 9之3D Touch

    金田 北京时间9月10日凌晨, Apple在美国旧金山比尔格拉汉姆公民大礼堂(Bill Graham Civic Auditorium)召开新品发布会.本次着重介绍了3D Touch功能, 大体介绍一 ...

  10. 【转】Android adb shell操作时出现“ XXX ... Read-only file system”解决办法--不错

    原文网址:http://blog.csdn.net/whu_zhangmin/article/details/25364349 手机连接PC后 adb shell su rm -r /system/a ...