题目大意:给定一棵 N 个节点的有根树,1 号节点为根节点,叶子节点有点权,每条边有边权,每经过一条边都减去该边权,每经过一个节点都加上该点权,求在保证权值和为非负数的前提下最多能经过多少个叶子节点。

题解:\(dp[u][i]\) 表示在以 u 为根节点的子树中,经过 i 个叶子节点的最大权值和,则有状态转移方程:$$dp[u][i]=max(dp[u][i],dp[v][k]+dp[u][i-k])$$。

一般前提为第一要素,作为要最优化的值,将要求的最优化的值最为附加属性,最后在满足前提的条件下遍历附加属性求出答案。

update on 2019.5.24

学习到了树上背包问题的上下界优化。

一开始做这道题肯定会觉得复杂度分析很奇怪,即:三个 for 循环竟然过了3000的数据量。

最后看了大佬的博客终于明白了,复杂度从严格意义上来说就是 \(O(n^2)\) 的。

感性证明如下:

我们实现的 dfs 过程可以看作是子树维护的信息合并的过程。在这个过程中,发现任意两个点为根节点的子树信息均发生且仅发生了一次合并。而对于任意两个点的信息合并仅发生在这两个节点的 lca 处,因此时间复杂度为 \(O(n^2)\)。

稍微严谨一点的证明如下:

\[T_u=\sum\limits_{fa[v]=u}T_v+f(u)
\]

首先证明对于子树合并的过程复杂度是 \(O(sz[u]^2)\) :

\[\begin{aligned} f_{u} &=1+\left(1+s i z\left[v_{1}\right]\right) \times \operatorname{siz}\left[v_{1}\right]+\left(1+\operatorname{siz}\left[v_{1}\right]+s i z\left[v_{2}\right]\right) \times \operatorname{siz}\left[v_{2}\right]+\cdots+\operatorname{siz}[u] \times \operatorname{siz}\left[v_{k}\right] \\ &\le 1+\sum_{fa\left[v_{i}\right]=u} \operatorname{siz}\left[v_{i}\right] \times(\operatorname{siz}[u]+1) \\ &=O(\operatorname{siz}[u]^{2}) \end{aligned}
\]

再证明前面子树的和式也是 \(O(sz[u]^2)\):

利用数学归纳法可知,每个子树都是 \(O(sz[v]^2)\) 的,那么在合并的过程中,利用均值不等式(平方的和大于和的平方)可直接证出。

因此,总的时间复杂度为 \(O(n^2)\)。

代码如下

#include <bits/stdc++.h>
using namespace std;
const int maxn=3010; struct node{
int nxt,to,w;
}e[maxn<<1];
int tot,head[maxn];
inline void add_edge(int from,int to,int w){
e[++tot]=node{head[from],to,w},head[from]=tot;
} int n,m,val[maxn],dp[maxn][maxn]; void read_and_parse(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n-m;i++){
int k;scanf("%d",&k);
for(int j=1;j<=k;j++){
int to,w;scanf("%d%d",&to,&w);
add_edge(i,to,w);
}
}
for(int i=n-m+1;i<=n;i++)scanf("%d",&val[i]);
memset(dp,0xcf,sizeof(dp));
} int dfs(int u){
dp[u][0]=0;
if(u>n-m){dp[u][1]=val[u];return 1;}
int sum=0;
for(int i=head[u];i;i=e[i].nxt){
int v=e[i].to;
int t=dfs(v);sum+=t;
for(int j=sum;j;j--)
for(int k=1;k<=t;k++)
dp[u][j]=max(dp[u][j],dp[u][j-k]+dp[v][k]-e[i].w);
}
return sum;
} void solve(){
dfs(1);
int ans=0;
for(int i=m;i;i--)if(dp[1][i]>=0){ans=i;break;}
printf("%d\n",ans);
} int main(){
read_and_parse();
solve();
return 0;
}

【洛谷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. HAProxy 日志输出及配置

    正所谓,没有软件敢说没有bug,人无完人,software is  not perfect software.是软件就可能存在bug,那么如果出现bug,我们就要分析对我们业务的影响及可能如何避免bu ...

  2. 《DISTRIBUTED SYSTEMS Concepts and Design》读书笔记 一

    第二章 系统模型 描述分布式系统的三种模型 Physical models : 用机器,网络,硬件等语言去描述整个系统. Architectural models : 用计算.计算任务.计算单元等语言 ...

  3. MATLAB找不到遗传算法工具箱,用不了gatool命令的解决方案

    解决方案 官方解释如下: gatool was removed as of R2015b. Use optimtool 在MATLAB R2015b前的版本可以使用gatool调用遗传算法工具箱,我测 ...

  4. PEP8 Python编程规范

    官方文档: https://www.python.org/dev/peps/pep-0008/ ---------------------------------------------------- ...

  5. 12.8 Daily Scrum

    最近大家都比较忙,任务今天也才刚刚分配,所以具体的编码任务从明天开始.   Tomorrow's Task 丁辛 完善餐厅列表,显示距离.             邓亚梅          美化搜索框 ...

  6. 转角遇见——Software

    第一部分:结缘计算机 从五岁开始读书,懵懵懂懂,从小就听长辈们说一定要考一个好大学,高三老师们就更是说:“过了高考,人生就无忧了”.于是似乎,高考就好像是我自出生以来这么多年的唯一愿景.高考成绩下来后 ...

  7. JavaScript中的cookie

    cookie本身没什么可介绍的,但是cookie在JavaScript中,有很多需要注意的 首先,cookie在JavaScript中,是window.document对象的一个属性,所以访问cook ...

  8. AWK学习一例

    awk 'BEGIN { for (i = 1; i <= 7; i++) print int(101 * rand()) }'

  9. Fiddler 跟踪手机请求.

    1. 想着跟踪下手机的部分请求, 所以使用fiddler 挂代理的方式来处理. 步骤 安装fiddler. 直接百度 安装即可.. 2. fiddler设置 tool-options设置 我试过解密h ...

  10. FICO基础知识(一)

    GL – 总账 (General Ledger) 总帐核算的中心任务是提供外部会计及其所涉及帐户的概貌. 总账会计主要用途:根据不同的会计准则(如欧洲的 IAS, 美国的GAAP, 中国国家会计准则) ...