类型:树形背包

传送门:>Here<

题意:给出一棵树,根节点在转播足球赛,每个叶子节点是一个观众在收看。每个叶子结点到根节点的路径权值之和是该点转播的费用,每个叶子节点的观众都会付val[i]的钱。先问在不亏本的前提下,最多转播多少观众

解题思路

$dp[u][i][j]$表示在以$u$为根节点的子树内,前$i$棵子树内选择$j$名观众的最大收益。这里的收益指的就是【所选观众交的钱 - 从$u$出发的转播费用总和】。从$u$出发就是指$u$以上的转播费用都不算

因此容易得到$dp[u][i][0]=0$(任何一个点不转播的收益都是0),为了一会儿打擂,其他权且置为$-∞$

在讲状态转移之前,先来回顾一下分组背包问题——物品已经分为几个组,题目要求每个组只能选择一个物品,使总价值最大。表面上看上去有点棘手,实际上就是一个01背包,只不过加了限定,一个组里只能选一个物品。因此正常的做法是遍历每一组,然后选择即可

那么当前这个问题可以转化为分组背包模型——每棵子树为一组,每组只能选择一个"物品"。这里的一组里的每个物品也就是"1个观众,2个观众,……"

因此易得转移方程$$dp[u][i][j]=Max\{dp[u][i-1][j-k] + dp[v][numson[v]][k] - cost[u \rightarrow v]\}$$注意要减去到达$v$的这一条边的费用。其中$numson[v]$是指$v$的直接儿子数量

观察可以发现$i$这一维可以滚动,因此可以优化为$$dp[u][j]=Max\{dp[u][j-k] + dp[v][k] - cost[u \rightarrow v]\}$$但是注意$j$要由$j-k$转移,因此$j$应当倒着扫。那$numson[v]$这一维可以舍去?因为$v$已经做完了,所以自然指的就是$numson[v]$了。

至于答案输出,我们只需要看$dp[1][q]$中满足$dp[1][q] \geq 0$的最大$q$就好了

Code

读入卡了我半天,这题的读入真奇特

/*By DennyQi 2018.8.14*/
#include <cstdio>
#include <queue>
#include <cstring>
#include <algorithm>
#define r read()
#define Max(a,b) (((a)>(b)) ? (a) : (b))
#define Min(a,b) (((a)<(b)) ? (a) : (b))
using namespace std;
typedef long long ll;
const int MAXN = ;
const int MAXM = ;
const int INF = ;
inline int read(){
int x = ; int w = ; register int c = getchar();
while(c ^ '-' && (c < '' || c > '')) c = getchar();
if(c == '-') w = -, c = getchar();
while(c >= '' && c <= '') x = (x << ) + (x << ) + c - '', c = getchar(); return x * w;
}
int N,M,R,x,y,K,ans;
int first[MAXM],nxt[MAXM],to[MAXM],cost[MAXM],num_edge;
int val[MAXN],dp[MAXN][MAXN],size[MAXN];
inline void add(int u, int v, int w){
to[++num_edge] = v,cost[num_edge] = w,nxt[num_edge] = first[u],first[u] = num_edge;
}
int DP(int u, int _f){
int v,isleaf=,tot=;
if(u > N-M){ dp[u][] = val[u]; return ; }
for(int i = first[u]; i; i = nxt[i]){
if((v = to[i]) == _f) continue;
tot += DP(v, u);
for(int j = tot; j; --j)
for(int k = ; k <= j; ++k)
dp[u][j] = Max(dp[u][j], dp[u][j-k] + dp[v][k] - cost[i]);
}
return tot;
}
int main(){
memset(dp, -0x3f, sizeof(dp));
N = r, M = r;
for(int i = ; i <= N-M; ++i){
K = r;
for(int j = ; j <= K; ++j){
x = r, y = r;
add(i, x, y), add(x, i, y);
}
}
for(int i = ; i <= N; ++i) dp[i][] = ;
for(int i = N-M+; i <= N; ++i) val[i] = r;
DP(, -);
for(int i = ; i <= M; ++i) if(dp[][i] >= ) ans = Max(ans, i);
printf("%d", ans);
return ;
}

[洛谷P1273] 有线电视网的更多相关文章

  1. 洛谷 P1273 有线电视网

    2016-05-31 13:25:45 题目链接: 洛谷 P1273 有线电视网 题目大意: 在一棵给定的带权树上取尽量多的叶子节点,使得sigma(val[选择的叶子节点])-sigma(cost[ ...

  2. 洛谷 P1273 有线电视网(树形背包)

    洛谷 P1273 有线电视网(树形背包) 干透一道题 题面:洛谷 P1273 本质就是个背包.这道题dp有点奇怪,最终答案并不是dp值,而是最后遍历寻找那个合法且最优的\(i\)作为答案.dp值存的是 ...

  3. 洛谷P1273 有线电视网 (树上分组背包)

    洛谷P1273 有线电视网 题目描述 某收费有线电视网计划转播一场重要的足球比赛.他们的转播网和用户终端构成一棵树状结构,这棵树的根结点位于足球比赛的现场,树叶为各个用户终端,其他中转站为该树的内部节 ...

  4. 【题解】洛谷P1273 有线电视网(树上分组背包)

    次元传送门:洛谷P1273 思路 一开始想的是普通树形DP 但是好像实现不大好 观摩了一下题解 是树上分组背包 设f[i][j]为以i为根的子树中取j个客户得到的总价值 我们可以以i为根有j组 在每一 ...

  5. 洛谷——P1273 有线电视网

    P1273 有线电视网 题目大意: 题目描述 某收费有线电视网计划转播一场重要的足球比赛.他们的转播网和用户终端构成一棵树状结构,这棵树的根结点位于足球比赛的现场,树叶为各个用户终端,其他中转站为该树 ...

  6. C++ 洛谷 P1273 有线电视网 题解

     P1273 有线电视网  很明显,这是一道树形DP(图都画出来了,还不明显吗?) 未做完,持续更新中…… #include<cstdio> #include<cstring> ...

  7. 洛谷P1273 有线电视网 树上分组背包DP

    P1273 有线电视网 )逼着自己写DP 题意:在一棵树上选出最多的叶子节点,使得叶子节点的值 减去 各个叶子节点到根节点的消耗 >= 0: 思路: 树上分组背包DP,设dp[u][k] 表示 ...

  8. 洛谷P1273 有线电视网 【树上分组背包】

    题目描述 某收费有线电视网计划转播一场重要的足球比赛.他们的转播网和用户终端构成一棵树状结构,这棵树的根结点位于足球比赛的现场,树叶为各个用户终端,其他中转站为该树的内部节点. 从转播站到转播站以及从 ...

  9. 洛谷P1273 有线电视网【树形dp】

    题目:https://www.luogu.org/problemnew/show/P1273 题意:一棵树,叶子节点是用户,每天边有一个权值表示花费,每一个用户有一个值表示他们会交的钱. 问在不亏本的 ...

  10. 洛谷 P1273 有线电视网(dp)

    /* 想了半天没想出状态 自己还是太弱了 QAQ 题目问的是最多供给多少户 一般想法是把这个值定义为状态量 没想出来QAQ....看了看题解的状态 很机智.... f[i][j]表示i的子树 选了j个 ...

随机推荐

  1. orleans发送广播消息

    一个client发送消息给orleans, 就只需要掉用Grain的函数就行了. 但是有时候Grain需要发送消息给client, 在orleans里面, 就只能通过Observer来实现. publ ...

  2. 设计模式之单例模式(C#)

    本文来自于本人个人微信公众号,欢迎关注本人微信公众号,二维码附在文章末尾~~~ 一直都特别羡慕能写文章的人,但是由于本人比较懒再加上写文章功底实在是just so so,所以就一搁再搁,最近突然觉得自 ...

  3. Python_迭代器-生成器-复习-习题_41

    # 迭代器和生成器# 迭代器 # 可迭代协议 —— 含有iter方法的都是可迭代的 # 迭代器协议 —— 含有next和iter的都是迭代器 # 特点 # 节省内存空间 # 方便逐个取值,一个迭代器只 ...

  4. POJ - 2528 区间离散化,线段树区间修改,区间询问

    这个题非常有意思的地方是,我们发现区间[1,4]和[5,8]是紧挨着的,因为这个的数代表的是一段区间,原本我们对于普通的离散, a[1]=1,a[2]=5,a[3]=6,a[4]=8;数组下标就是重新 ...

  5. python 中的re模块,正则表达式

    一.re模块 re模块中常用的方法. match: 默认从字符串开头开始匹配,re.match('fun', 'funny') 可以匹配出来 'fun' match(pattern, string, ...

  6. 【问题解决方案】之 关于某江加密视频swf专用播放器仍无法播放的问题

    前言: 从pt上下载了一些语言学习的视频之后一直打不开,百度谷歌了若干种方法仍然无解.无奈放弃. 某日从百度知道里又看到一个方法,试了一下,居然灵了.呜呼哀哉.赶紧记下来. 原方法链接:https:/ ...

  7. JEECG 3.7 Memory Leak

    JEECG 3.7 版本常见问题贴 - JEECG官方网站-企业级JAVA快速开发平台 - Powered by Discuz!http://www.jeecg.org/forum.php?mod=v ...

  8. TCP 握手和挥手图解(有限状态机)

    1.引言 TCP 这段看过好几遍,老是记不住,没办法找工作涉及到网络编程这块,各种问 TCP .今天好好整理一下握手和挥手过程.献给跟我一样忙碌,找工作的童鞋,欢迎大神批评指正. 2.TCP 的连接建 ...

  9. [转帖]关于CP936

    来源: 知乎:https://www.zhihu.com/question/35609295/answer/63780022 CP936和UTF-8本身和Python是毫无关联的. CP936其实就是 ...

  10. GANs (Generative Adversarial Networks)

    #!/usr/bin/python2.7 #coding:utf-8 import tensorflow as tf import numpy as np import matplotlib.pypl ...