P1273 有线电视网

题面

题目描述

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

从转播站到转播站以及从转播站到所有用户终端的信号传输费用都是已知的,一场转播的总费用等于传输信号的费用总和。

现在每个用户都准备了一笔费用想观看这场精彩的足球比赛,有线电视网有权决定给哪些用户提供信号而不给哪些用户提供信号。

写一个程序找出一个方案使得有线电视网在不亏本的情况下使观看转播的用户尽可能多。

输入输出格式

输入格式:

输入文件的第一行包含两个用空格隔开的整数 \(N\) 和 \(M\) ,其中 \(2 \leq N \leq 3000\) , \(1 \leq M \leq N-1\) , \(N\) 为整个有线电视网的结点总数, \(M\) 为用户终端的数量。

第一个转播站即树的根结点编号为 \(1\) ,其他的转播站编号为 \(2\) 到 \(N-M\) ,用户终端编号为 \(N-M+1\) 到 \(N\) 。

接下来的 \(N-M\) 行每行表示—个转播站的数据,第 \(i+1\) 行表示第 \(i\) 个转播站的数据,其格式如下:

\[K \ A_1 \ C_1 \ A_2 \ C_2 … A_k \ C_k
\]

\(K\) 表示该转播站下接 \(K\) 个结点(转播站或用户),每个结点对应一对整数 \(A\) 与 \(C\) , \(A\) 表示结点编号, \(C\) 表示从当前转播站传输信号到结点 \(A\) 的费用。最后一行依次表示所有用户为观看比赛而准备支付的钱数。

输出格式:

输出文件仅一行,包含一个整数,表示上述问题所要求的最大用户数。

输入输出样例

输入样例:

  1. 5 3
  2. 2 2 2 5 3
  3. 2 3 2 4 3
  4. 3 4 2

输出样例:

  1. 2

说明

样例解释

如图所示,共有五个结点。结点 \(①\) 为根结点,即现场直播站, \(②\) 为一个中转站, \(③④⑤\) 为用户端,共 \(M\) 个,编号从 \(N-M+1\) 到 \(N\) ,他们为观看比赛分别准备的钱数为 \(3, 4, 2\) ,从结点 \(①\) 可以传送信号到结点 \(②\) ,费用为 \(2\) ,也可以传送信号到结点 \(⑤\) ,费用为 \(3\) (第二行数据所示),从结点 \(②\) 可以传输信号到结点 \(③\) ,费用为 \(2\) 。也可传输信号到结点 \(④\) ,费用为 \(3\) (第三行数据所示),如果要让所有用户( \(③④⑤\) )都能看上比赛,则信号传输的总费用为:

\(2+3+2+3=10\) ,大于用户愿意支付的总费用 \(3+4+2=9\) ,有线电视网就亏本了,而只让 \(③④\) 两个用户看比赛就不亏本了。

思路

肝试炼场咯! -Uranus

设 \(dp[i][j]\) 表示在 \(i\) 节点满足 \(j\) 位观众的最大收益,那么我们就可以用 \(i\) 节点的每一个儿子节点来更新。

\[dp[i][j]=max \{ dp[i][j-k]+dp[w][k]-cost[i][w](w \in son[i]) \}
\]

然后就在 \(dfs\) 的过程中做这样的动态规划就好啦。

AC代码

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. const int MAXN=3005;
  4. const int INF=1000000;
  5. int n,m,ans,cnt[MAXN],sz[MAXN],son[MAXN][MAXN],c[MAXN][MAXN],dp[MAXN][MAXN];
  6. int read()
  7. {
  8. int re=0;
  9. char ch=getchar();
  10. while(!isdigit(ch)) ch=getchar();
  11. while(isdigit(ch)) re=(re<<3)+(re<<1)+ch-'0',ch=getchar();
  12. return re;
  13. }
  14. void dfs(int now)
  15. {
  16. for(int i=1;i<=cnt[now];i++)
  17. {
  18. dfs(son[now][i]);
  19. sz[now]+=sz[son[now][i]];
  20. for(int k=sz[now];k>=0;k--)
  21. for(int j=0;j<=sz[son[now][i]];j++)
  22. dp[now][k]=max(dp[now][k],dp[now][k-j]+dp[son[now][i]][j]-c[now][i]);
  23. }
  24. }
  25. int main()
  26. {
  27. n=read(),m=read();
  28. for(int i=1;i<=n;i++)
  29. for(int j=1;j<=m;j++)
  30. dp[i][j]=-INF;
  31. for(int i=1;i<=n-m;i++)
  32. {
  33. cnt[i]=read();
  34. for(int j=1;j<=cnt[i];j++) son[i][j]=read(),c[i][j]=read();
  35. }
  36. for(int i=n-m+1;i<=n;i++) sz[i]=1,dp[i][1]=read();
  37. dfs(1);
  38. for(int i=m;i>=0;i--)
  39. if(dp[1][i]>=0)
  40. {
  41. printf("%d",i);
  42. return 0;
  43. }
  44. }

Luogu P1273 有线电视网(树形dp+背包)的更多相关文章

  1. Luogu P1273 有线电视网 树形DP

    又重构了一下...当然当初的题一看就看懂了QAQ 设f[i][j]表示以i为根的子树,有j个客户的最大收益 方程:f[u][j+k]=max(f[u][j+k],f[u][j]+f[v][k]-w(u ...

  2. P1273 有线电视网(树形dp)

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

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

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

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

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

  5. Luogu P1273 有线电视网

    最近写DP写得比较多了 但是POJ上的题目太傻比了而且不想看英文的题面,然后就在Luogu的试炼场里找了一个DP EX专题写了一下(大概3days吧,一天一题差不多) 这是一道比较简单的DP 话说树形 ...

  6. luoguP1273 有线电视网 [树形dp]

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

  7. Luogu 1273 有线电视网 - 树形背包

    Description 树形背包, 遍历到一个节点, 枚举它的每个子节点要选择多少个用户进行转移. Code #include<cstring> #include<cstdio> ...

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

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

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

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

随机推荐

  1. C# 字符串的日期比较

    SearchResult = SearchResult.Where(v => (DateTime.Parse(v.CreateTime.ToString("yyyy/MM/dd&quo ...

  2. css3 ---1 基本的选择器

    基本的选择器 <style type="text/css"> /*通配符选择器*/ * { margin: ; padding: ; border: none; } / ...

  3. html-from提交表单

    使用form创建的仅仅是一个空白的表单, 我们还需要向form中添加不同的表单项 <!DOCTYPE html> <html> <head> <meta ch ...

  4. 浏览器禁用Cookie后的Session处理

    1. 实现购物车, 可以基于Cookie, 也可以基于Session, 若服务器性能较差, 可以考虑基于Cookie实现购物车 2. 解决方案: URL重写 把用户可能点的每一个超链接后面,都跟上用户 ...

  5. PHP面向对象魔术方法基本了解

    简单介绍 (1) 魔术方法都是系统提供,程序员使用即可. (2) 所有的魔术方法,前面都是以  __ 开头的 _是两个下划线. (3) 我们在自定义函数时,就不要使用 __开头了. (4) 魔术方法是 ...

  6. String类的trim() 方法

    用于删除字符串的头尾空白符. 语法:public String trim() 返回值:删除头尾空白符的字符串. public class Test {    public static void ma ...

  7. 黑裙晖安装后修改mac和sn

    d当前使用6.2 打开putty sudo -i 然后在/tmp目录下创建一个临时目录,名字随意,如:boot mkdir -p /tmp/boot 第四步:切换到dev目录 cd /dev 第五步: ...

  8. 用python打造简单的cms识别

    代码 #!/usr/bin/env python3 # coding:utf-8 #lanxing #判断代码,判断是否安装requests库 try: import requests except: ...

  9. 查看MySql版本号命令

    转自:https://blog.csdn.net/qq_38486203/article/details/80324014​ 这里介绍四中不同的方法,它们分别运行在不同的环境中,最后对每种方法的优劣以 ...

  10. 跟我一起安装vmware

    第一步查看我们的电脑配置 我是windows10,下面的方法是windows10来安装vmware 第二步双击下图文件 (1) 2)弹出如下图,点击下一步即可. (3)一直点击下一步(期间会同意,勾选 ...