洛谷P4017 最大食物链计数

  这是洛谷一题普及/提高-的题目,也是我第一次做的一题 图上动态规划/拓扑排序 ,我认为这题是很好的学习拓扑排序的题目。

    在这题中,我学到了几个名词,入度,出度,及没有环的有向图必定有入度为0的点。通过与题干分析可知,入度为0就是最佳生产者,出度为0就是最佳消费者。题干的大意就是找出图中一共有几条食物链是从最佳生产者指向最佳消费者。

    我在题解区学习了拓扑排序后的第一次题解,然而只过了一个测试点,一片WA声。。

 1 //动态规划 洛谷P4017 最大食物链计数
2 #include<iostream>
3 #include<cmath>
4 #include<vector>
5 #include<queue>
6 using namespace std;
7 const int mod = 80112002;
8 const int MAX = 5e3 + 5;
9 int in[MAX], out[MAX];//两个数组 记录结点的入度和出度
10 vector<int>neigh[MAX]; //vector的二维数组 可以存放图的邻接关系
11 queue<int>que;//队列 每次将入度为0的点入队循环
12 int num[MAX];//记录每一个点的值 就是到这个点有几条路径
13 int ans;
14 int main()
15 {
16 int n, m;
17 int x, y;//用来输入被捕食和捕食的生物编号
18 cin >> n >> m;//n是生物数量 m是食物链数量 也可以说n就是结点数 m就是有向边的数量
19 for (int i = 0; i < m; ++i)
20 {
21 cin >> x >> y;
22 out[x]++, in[y]++;//图上是由x-->y所以入度y++ 出度x++
23 neigh[x].push_back(y);//存放x指向的结点 便于后面遍历
24 }
25
26 for (int i = 1; i <= n; ++i)
27 {
28 //遍历寻找入度为0的点 也就是最佳生产者 加入队列
29 if (!in[i])
30 {
31 num[i] = 1;//将入度为0的点的num设为1 自己到自己有一条路径
32 que.push(i);
33 break;
34 }
35 }
36 int temp;
37 while (!que.empty())
38 {
39 temp = que.front();
40 que.pop();
41 //遍历这个结点所有连接的结点
42 for (int i = 0; i < neigh[temp].size(); ++i)
43 {
44 //把这些连接点的入度都减1 并且让他们的num值都加上前一个结点的num值
45 in[neigh[temp][i]]--;
46 num[neigh[temp][i]] = (num[temp] + num[neigh[temp][i]]) % mod;
47 if (in[neigh[temp][i]] == 0)
48 que.push(neigh[temp][i]);//如果入度为0 就把这个结点加入队列
49 }
50 }
51 for (int i = 1; i <= n; ++i)
52 {
53 //寻找出度为0的结点 出度为0说明就是最佳消费者 也就是最后一个终点结点
54 if (out[i] == 0)
55 {
56 ans = num[i];
57 break;
58 }
59
60 }
61 cout << ans;
62
63 return 0;
64
65 }

经过我的思考与debug,我发现,不一定只有一个最佳生产者,也不一定只有一个最佳消费者,所以起点和终点应该有很多个,所以ans应该累加,起始队列的添加应该遍历完数组,可能添加多个起点。

修改后的代码:

 1 //动态规划 洛谷P4017 最大食物链计数
2 #include<iostream>
3 #include<cmath>
4 #include<vector>
5 #include<queue>
6 using namespace std;
7 const int mod = 80112002;
8 const int MAX = 5e3 + 5;
9 int in[MAX], out[MAX];//两个数组 记录结点的入度和出度
10 vector<int>neigh[MAX]; //vector的二维数组 可以存放图的邻接关系
11 queue<int>que;//队列 每次将入度为0的点入队循环
12 int num[MAX];//记录每一个点的值 就是到这个点有几条路径
13 int ans;
14 int main()
15 {
16 int n, m;
17 int x, y;//用来输入被捕食和捕食的生物编号
18 cin >> n >> m;//n是生物数量 m是食物链数量 也可以说n就是结点数 m就是有向边的数量
19 for (int i = 0; i < m; ++i)
20 {
21 cin >> x >> y;
22 out[x]++, in[y]++;//图上是由x-->y所以入度y++ 出度x++
23 neigh[x].push_back(y);//存放x指向的结点 便于后面遍历
24 }
25
26 for (int i = 1; i <= n; ++i)
27 {
28 //遍历寻找入度为0的点 也就是最佳生产者 加入队列
29 if (!in[i])
30 {
31 num[i] = 1;//将入度为0的点的num设为1 自己到自己有一条路径
32 que.push(i);//入度为0的点有很多个
33
34 }
35 }
36 int temp;
37 while (!que.empty())
38 {
39 temp = que.front();
40 que.pop();
41 //遍历这个结点所有连接的结点
42 for (int i = 0; i < neigh[temp].size(); ++i)
43 {
44 //把这些连接点的入度都减1 并且让他们的num值都加上前一个结点的num值
45 in[neigh[temp][i]]--;
46 num[neigh[temp][i]] = (num[temp] + num[neigh[temp][i]]) % mod;
47 if (in[neigh[temp][i]] == 0)
48 que.push(neigh[temp][i]);//如果入度为0 就把这个结点加入队列
49 }
50 }
51 for (int i = 1; i <= n; ++i)
52 {
53 //寻找出度为0的结点 出度为0说明就是最佳消费者 也就是最后一个终点结点
54 if (out[i] == 0)
55 {
56 ans = (ans + num[i]) % mod;
57 //出度为0的点也可能有很多个 不能break 要累加答案
58 }
59
60 }
61 cout << ans;
62
63 return 0;
64
65 }

完美通过

我们可以从这题中学习到拓扑排序的一个模板。

入度in[N] 出度out[N]的初始化数值
通过二维vector<int> neigh[N]来存储邻接图
num[N]来存储每个结点的答案数值
遍历入度来找到入度为0的点添加至队列queue<int>q
遍历队列的基本模板
while (!que.empty())
{
temp = que.front();
que.pop();
//遍历这个结点所有连接的结点
for (int i = 0; i < neigh[temp].size(); ++i)
{
//把这些连接点的入度都减1 并且让他们的num值都加上前一个结点的num值
in[neigh[temp][i]]--;
num[neigh[temp][i]] = (num[temp] + num[neigh[temp][i]]) % mod;
if (in[neigh[temp][i]] == 0)
que.push(neigh[temp][i]);//如果入度为0 就把这个结点加入队列
}
}
最后遍历out数组找到出度为0的点 num值累加为ans即为但答案。

动态规划 洛谷P4017 最大食物链计数——图上动态规划 拓扑排序的更多相关文章

  1. 洛谷 P4017 最大食物链计数

    洛谷 P4017 最大食物链计数 洛谷传送门 题目背景 你知道食物链吗?Delia生物考试的时候,数食物链条数的题目全都错了,因为她总是重复数了几条或漏掉了几条.于是她来就来求助你,然而你也不会啊!写 ...

  2. 洛谷 P4017 最大食物链计数 题解

    P4017 最大食物链计数 题目背景 你知道食物链吗?Delia生物考试的时候,数食物链条数的题目全都错了,因为她总是重复数了几条或漏掉了几条.于是她来就来求助你,然而你也不会啊!写一个程序来帮帮她吧 ...

  3. 洛谷——P4017 最大食物链计数

    P4017 最大食物链计数 题目背景 你知道食物链吗?Delia生物考试的时候,数食物链条数的题目全都错了,因为她总是重复数了几条或漏掉了几条.于是她来就来求助你,然而你也不会啊!写一个程序来帮帮她吧 ...

  4. 洛谷 P4017 最大食物链计数 (拓扑排序,思维)

    题意:有\(n\)个点,连\(m\)条边,求最多有多少条食物链(从头走到为有多少条路径). 题解:之前抽了点时间把拓扑排序补完了,这题其实就是一道拓扑排序的裸题.关于拓扑排序: ​ 1.首先,我们用\ ...

  5. 洛谷P4017 最大食物链计数

    拓扑排序板子题 #include <iostream> #include <cstdio> #include <cstring> #include <queu ...

  6. 洛谷P4581 [BJOI2014]想法(玄学算法,拓扑排序)

    洛谷题目传送门 萝卜大毒瘤 题意可以简化成这样:给一个DAG,求每个点能够从多少个入度为\(0\)的点到达(记为\(k\)). 一个随机做法:给每个入度为\(0\)的点随机一个权值,在DAG上求出每个 ...

  7. 洛谷P4645 [COCI2006-2007 Contest#7] BICIKLI [Tarjan,拓扑排序]

    题目传送门 BICIKLI 题意翻译 给定一个有向图,n个点,m条边.请问,1号点到2号点有多少条路径?如果有无限多条,输出inf,如果有限,输出答案模10^9的余数. 两点之间可能有重边,需要看成是 ...

  8. 洛谷P4017 最大食物链数量 dfs

    老规矩,传送门 做题从头到尾的思路: 1. 这个题明显就是dfs数数量了,简单,邻接矩阵干他! TLE警告,8个点 额... 2. 老师说这玩意不能邻接矩阵?没事,还有邻接表,再来! 再次TLE 8个 ...

  9. 洛谷P3065 [USACO12DEC]第一!First!(Trie树+拓扑排序)

    P3065 [USACO12DEC]第一!First! 题目链接:https://www.luogu.org/problemnew/show/P3065 题目描述 Bessie一直在研究字符串.她发现 ...

随机推荐

  1. 实例化类对象及类的属性set方法使用不当

    类的属性中set方法操作数据库,新建类对象并给其赋值时总会触发该set方法,而导致不期望的错乱: 库位类Storage,其中传感器状态SensorStatus和逻辑状态LogicStatus有一定的关 ...

  2. Jmeter--由PV估算tps和最大并发数

    需求 "假设一个系统的业务有登录.浏览帖子.发送新贴.回复帖子,访问高峰是上午10点,日访问高峰PV约5208(含登录1300.浏览2706.发帖526.回帖676).系统响应时间要求小于3 ...

  3. 表格的td行利用css显示...

      默认超过指定长度以...显示, 鼠标放到文本上显示全 代码如下 .fh{ max-width:220px; word-wrap:break-word; text-overflow:ellipsis ...

  4. laravel8 登录功能的实现

    1.选择合适的框架,渲染出如上图所示的登录视图,视图有样式即可,可使用BootStrap或layUI去布局实现(10分) 2.正确显示出验证码(10分) 3.验证码要求无杂点.无干扰线,4位纯数字(1 ...

  5. LGP3449题解

    其实每个串都不是回文串也能做的说... 题意:给定 \(n\) 个互不相同的串,两两拼接一共能够拼出 \(n^2\) 个串,问这 \(n^2\) 个串中有几个回文串. 首先假设拼接出来的串是 \(AB ...

  6. 盘点十大GIS相关算法

    1.道格拉斯-普克算法(Douglas–Peucker) 道格拉斯-普克算法(Douglas–Peucker algorithm,亦称为拉默-道格拉斯-普克算法.迭代适应点算法.分裂与合并算法)是将曲 ...

  7. Netty之DefaultAttributeMap与AttributeKey的机制和原理

    一.介绍和原理分析 1.什么是 DefaultAttributeMap? DefaultAttributeMap 是一个 数组 + 链表 结构的线程安全Map. 2.什么是 AttributeKey? ...

  8. 为什么你需要在用 Vue 渲染列表数据时指定 key

    本文改写整理自一篇博文,原文链接如下: Why you should use the key directive in Vue.js with v-for Application state and ...

  9. 阐述final、finally、finalize的区别?

    - final:修饰符(关键字)有三种用法:如果一个类被声明为final,意味着它不能再派生出新的子类,即不能被继承,因此它和abstract是反义词.将变量声明为final,可以保证它们在使用中不被 ...

  10. java-設計模式-工場方法

      工廠方法: 一种创建型设计模式, 其在父类中提供一个创建对象的方法, 允许子类决定实例化对象的类型. 定义一个创建产品对象的工厂接口,将产品对象的实际创建工作推迟到具体子工厂类当中. 这满足创建型 ...