hihoCoder 1175 拓扑排序
hihoCoder 1175 拓扑排序
描述
小Hi和小Ho所在学校的校园网被黑客入侵并投放了病毒。这事在校内BBS上立刻引起了大家的讨论,当然小Hi和小Ho也参与到了其中。从大家各自了解的情况中,小Hi和小Ho整理得到了以下的信息:
- 校园网主干是由N个节点(编号1…N)组成,这些节点之间有一些单向的网路连接。若存在一条网路连接(u,v)链接了节点u和节点v,则节点u可以向节点v发送信息,但是节点v不能通过该链接向节点u发送信息。
- 在刚感染病毒时,校园网立刻切断了一些网络链接,恰好使得剩下网络连接不存在环,避免了节点被反复感染。也就是说从节点i扩散出的病毒,一定不会再回到节点i。
- 当1个病毒感染了节点后,它并不会检查这个节点是否被感染,而是直接将自身的拷贝向所有邻居节点发送,它自身则会留在当前节点。所以一个节点有可能存在多个病毒。
- 现在已经知道黑客在一开始在K个节点上分别投放了一个病毒。
举个例子,假设切断部分网络连接后学校网络如下图所示,由4个节点和4条链接构成。最开始只有节点1上有病毒。
最开始节点1向节点2和节点3传送了病毒,自身留有1个病毒:
其中一个病毒到达节点2后,向节点3传送了一个病毒。另一个到达节点3的病毒向节点4发送自己的拷贝:
当从节点2传送到节点3的病毒到达之后,该病毒又发送了一份自己的拷贝向节点4。此时节点3上留有2个病毒:
最后每个节点上的病毒为:
小Hi和小Ho根据目前的情况发现一段时间之后,所有的节点病毒数量一定不会再发生变化。那么对于整个网络来说,最后会有多少个病毒呢?
提示:拓扑排序的应用
小Hi:对于这个问题小Ho你有什么想法么?
小Ho:有,对于一个病毒来说它总会传递到一个没有邻居的节点,那我直接使用dfs模拟整个过程不就能够得到结果了吗?
小Hi:小Ho你真聪明,立刻就看穿了这个问题的本质。
小Ho:那当然啦。 <得意>
小Hi:那么小Ho,我这里有这样一个网络:
这里一共有n个节点,对于节点i,它总是连接着节点i+1和节点i+2。一开始只有节点1被感染,你算算最后n个节点一共有多少个病毒?
小Ho:很显然,节点1最后只有1个病毒;节点2也只有1个病毒;节点3会接受从节点1和节点2过来的病毒,所以有2个病毒;后面依次是节点4有3个病毒,节点5有5个病毒…它们好像刚好是费波拉契数列?
小Hi:你说的没错。对于这个网络,编号为i的节点最后感染的病毒数量就是斐波拉契数列的第i项。而斐波拉契数列的增长是很惊人的,当i到达一定值时,其电脑感染的病毒数量就会很大了。
小Ho:那这又有什么关系呢?反正能得到解不是么?
小Hi:能得到解那是当然了,但是你有想过需要花费多少时间么?在你的dfs算法中,每一次进入函数就等于模拟一个病毒进入电脑。如果最后结果有10亿个病毒,那么你就需要执行10亿次函数,假设电脑每秒钟可以执行1亿次函数,那么你的dfs就需要运行10秒。如果结果有100亿,1000亿呢?
小Ho:运行时间上好像是有问题,那我应该如何解决时间的问题呢?
小Hi:其实也很简单啦。你再想想,在我们最后得到结果上每各节点病毒数量有什么关系么?
小Ho:嗯 <思考> …我发现了!对于节点i来说,它最后的病毒数好像总是等于所有能够达到它的节点病毒数之和。就用你提到的那个斐波拉契数列来说,能够达到节点i的节点是节点i-1和节点i-2,所以节点i的病毒数是节点i-1和节点i-2的病毒数之和。这刚好就是斐波拉契数列的递推公式嘛!
小Hi:对,正是这样。对于一个节点i来说,如果我们能够先计算出它所有前驱节点的病毒数量,就可以直接推算出它最后的病毒数量了。
小Ho:但是怎么来计算所有前驱节点呢?
小Hi:这就要从图的性质入手了。我们现在的网络是没有环的,对于任意一个节点i,当它将自己所有的病毒都传送出去之后,它自身的病毒数量就不会改变了。那么我们不妨从没有前驱节点,也就是入度为0的节点开始考虑。
对于这些节点,它并不会再增加病毒数量。那么我们就根据它所关联的连接将病毒分发出去,然后这个节点就没有作用了。那不妨就删掉好了,它所关联的边也删掉。
这样图中又会产生一些新的没有入度的节点。这样一直删点,直到所有的点都被删掉,将所有点的病毒数量加起来不就是总的病毒数么?
我们不妨来看个例子,这里给定一个网络:
最开始只有节点1是入度为0的点,所以将它的病毒传送,然后删掉节点1:
此时节点2成为了入度为0的点,同样将其删掉:
此时节点3为入度为0的点,同样操作:
最后只剩下节点4,因为它并没有后续节点,所以病毒感染的过程也就结束了。
小Ho:这不就是拓扑排序么!因为拓扑排序可以在O(n+m)的时间解决,整个问题的时间也减少到了O(n+m),那么再多的节点数都不怕了!
小Hi:没错,这样我们就完美地解决了这个问题。你能实现它么?
小Ho:没问题,就交给我吧! <自信>
Close
输入
第1行:3个整数N,M,K,1≤K≤N≤100,000,1≤M≤500,000
第2行:K个整数A[i],A[i]表示黑客在节点A[i]上放了1个病毒。1≤A[i]≤N
第3…M+2行:每行2个整数 u,v,表示存在一条从节点u到节点v的网络链接。数据保证为无环图。1≤u,v≤N
输出
第1行:1个整数,表示最后整个网络的病毒数量 MOD 142857
Sample Input
4 4 1
1
1 2
1 3
2 3
3 4
Sample Output
6
一.题目分析
本题是典型的拓扑排序类题目 , 本题既可以用队列的方法求解也可以用数组模拟栈的方法求解
二.AC代码
using namespace std;
#include<bits/stdc++.h>
#define maxn 1000005
#define modd 142857
void f(void);
int sum;
int n,m,k,aa;
int a[maxn];
struct edge
{int s;//边的头节点int v;//边的尾节点int next1;// 以第i条边的头结点为起始点的上一条边的序号
}b[maxn];//保存边的信息
int head[maxn];//head[i] 表示以结点i为起始点的 最后给出的边的序号
int top1;
int in[maxn];//存储入度
int stack1[maxn];//模拟栈int main()
{scanf("%d%d%d",&n,&m,&k);for(int i=1;i<=k;i++){scanf("%d",&aa);a[aa]=1;}for(int i=1;i<=m;i++){int x,y;scanf("%d%d",&x,&y);b[i].s=x;//存储该条边的信息b[i].v=y;//存储该条边的信息b[i].next1=head[x];//该条边的下一条边的序号是 与该边同起点的上一条边的序号head[x]=i;//更新以该顶点为起点的 最后一条边的序号}f();for(int i=1;i<=n;i++){sum=(a[i]+sum)%modd;//注意对于这种取模的部分应写成sum=(a[i]+sum)%modd;而不要写成sum+=a[i]%modd;}printf("%d\n",sum);//对于sum最好不用长整型,长整型的取模跟int型的可能不太一样
}void f(void)
{for(int i=1;i<=n;i++){for(int j=head[i];j!=0;j=b[j].next1){in[b[j].v]++;}}int top1=-1;for(int i=1;i<=n;i++){if(in[i]==0) stack1[++top1]=i;//入度为0的结点}while(top1!=-1){int index=stack1[top1--];//取到当前入度为0的结点for(int i=head[index];i!=0;i=b[i].next1){if(--in[b[i].v]==0) stack1[++top1]=b[i].v;//减少以当前结点为头的结点的入度a[b[i].v]=(a[b[i].v]+a[index])%modd;}}
}
hihoCoder 1175 拓扑排序相关推荐
- HihoCoder - 1175 拓扑排序·二
描述 小Hi和小Ho所在学校的校园网被黑客入侵并投放了病毒.这事在校内BBS上立刻引起了大家的讨论,当然小Hi和小Ho也参与到了其中.从大家各自了解的情况中,小Hi和小Ho整理得到了以下的信息: 校园 ...
- HihoCoder - 1174 拓扑排序·一
由于今天上课的老师讲的特别无聊,小Hi和小Ho偷偷地聊了起来. 小Ho:小Hi,你这学期有选什么课么? 小Hi:挺多的,比如XXX1,XXX2还有XXX3.本来想选YYY2的,但是好像没有先选过YYY ...
- 题解报告:hihoCoder #1175:拓扑排序·二
题目链接:https://hihocoder.com/problemset/problem/1175 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho所在学 ...
- hihocoder 1343 : Stable Members【拓扑排序】
hihocoder #1343:题目 解释: 一个学习小组,一共有N个学员,一个主管.每个学员都有自己的导师(一个或者多个),导师可以是其他学员也可以是主管. 每周学员都要把自己的学习报告和收到的报告 ...
- 【图论】有向无环图的拓扑排序
1. 引言 有向无环图(Directed Acyclic Graph, DAG)是有向图的一种,字面意思的理解就是图中没有环.常常被用来表示事件之间的驱动依赖关系,管理任务之间的调度.拓扑排序是对DA ...
- C#实现有向无环图(DAG)拓扑排序
对一个有向无环图(Directed Acyclic Graph简称DAG)G进行拓扑排序,是将G中所有顶点排成一个线性序列,使得图中任意一对顶点u和v,若边(u,v)∈E(G),则u在线性序列中出现在 ...
- hdu 5438 Ponds 拓扑排序
Ponds Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/contests/contest_showproblem ...
- usaco frame up(所有拓扑排序的输出)
先根据图建图再拓扑排序. /** ID: jinbo wu TASK: frameup LANG:C++ */ #include<bits/stdc++.h> using namespac ...
- HDU1811 Rank of Tetris 拓扑排序+并查集 OR 差分约束最短路+并查集
题目链接 题意:就是给你一堆关系,看能不能排出个确定的顺序 做法: 1. 拓扑排序+并查集 应该很容易想到的一种思路,大于小于建立单向边.对于相等的呢,就把他们缩成一个点.就用并查集缩成一个点就行了 ...
最新文章
- 23、OSPF配置实验之特殊区域Totally NSSA
- Origin绘制大小和颜色可变的3D散点图
- php 微信开发 菜单,微信公众号中个性化菜单的开发实例
- RabbitMQ——无法连接错误[AmqpConnectException: java.net.ConnectException: Connection refused: connect]解决方案
- 我的世界光影mod怎么用_用“戏剧化”光影艺术 唤醒千年沉睡世界!
- confirm修改按钮文字_踏入MG动画设计的门,才知道文字动画这么重要……
- pythonfromlxml导入html_Python爬虫日记二:使用lxml解析HTML输出对应值
- LeetCode刷题(21)
- spring 事物的级别_Spring 事务中的隔离级别有哪几种?
- 「python自动化运维」之自动生成配置文件并上传到目标服务器指定的位置
- L298N模块详细使用教程
- pon移动家庭网关有虚拟服务器吗,电信、移动、联通家庭网关对比分析
- 【数据挖掘案例】财政收入影响因素分析及预测模型
- 中小银行传统数据仓库向大数据平台迁移探索
- 学习c语言有什么作用,c语言有什么用 小白如何学习c语言
- 大数据技术框架有哪些类型?大数据技术栈包括哪些框架?
- 完美解码播放器(PotPlayer)关闭自动播放下一首的方法(找了好久)
- EXT3文件系统修复
- Linux 中Mysql出现‘> 如何退出?
- 一寸照纯红色底图片_如何用最快的速度给证件照换底色?
热门文章
- UVa(12821),MCMF
- Eclipse 工具栏不见了
- androidannotations 在android studio中的使用
- 解决mysql“Access denied for user 'root'@'localhost'”
- linux 虚拟 设备,Linux中的虚拟网络设备
- rust电器元件需要什么材料_腐蚀Rust电力系统有哪些工具 腐蚀Rust电力系统工具汇总 其他工具-游侠网...
- Spring-@Value
- 计算机网络自顶向下-应用层
- graphviz 画决策树_决策树算法-实战篇
- git ssh配置文件 服务器_【GIT】日常开发中的这些Git技巧你知道吗?