@AkamemakA
2022-07-03T14:08:56.000000Z
字数 1915
阅读 327
Atcoder 题解
题目链接:点我
有个城镇,编号为。
有条路,第条路连接城镇和城镇。每条路是双向的。一个人经过一条路从一边走到另一边需要分钟。但对于某些道路,如果其,则代表这条路的一边连接着,但另一边不确定。
对于每一个,回答以下问题:
如果所有的有不确定城镇的道路都连接到城镇,那么一个人需要多少分钟才能从城镇走到城镇?输出这个时间。如果无法走到,则输出。
3 20 21 2
-1 -1 2
如果所有的未确定道路都连接到城镇,则道路和道路都连接了城镇和。是不可能从城镇走到城镇的。
如果所有的未确定道路都连接到城镇,则道路连接了号城镇自己,道路连接了城镇和,也是不可能从城镇走到城镇的。
如果所有的未确定道路都连接到城镇,则道路连接了城镇和,道路连接了城镇和,于是可以从用分钟城镇走到城镇:
因此输出为 。
请注意,确定了不确定的道路连接到的城镇,可能有一条道路连接城镇和城镇本身,也可能有多条道路连接同一对城镇。
5 51 21 33 44 50 2
3 3 3 3 2
一道美丽的图论题。
构造一个图,并且构造一个城镇。如果一条道路的一端连到了不确定的端点,那么就直接连上城镇。当我们确定这个不确定的城镇时,那么就相当于在城镇与城镇之间连接了一条长度为的虚拟边。
定义为城镇道城镇之间需要的时间。如果,则代表与之间不能到达。
当回答第个询问时,如果不需要经过城镇就可以从城镇走到城镇的话,那么一个备选答案就是。
否则,需要经过城镇的话,那么答案就为或者是。三者当中选最小值。
可以用下图作为参考。

由于我们在实现过程中,只会用到以和为起点的值,因此我们可以跑两次bfs来计算出城镇和的值。
#include<bits/stdc++.h>using namespace std;const int N=3e5+5,INF=0x3f3f3f3f; //INF表示正无穷,即两边不可到达int n,m;int he[N],ne[N<<1],go[N<<1],val[N<<1],tot; //前向星int vis[N],dis1[N],disn[N]; //vis为标记数组,dis1和disn分别表示从1和n开始的dis值inline void add(int a,int b,int c){ne[++tot]=he[a];he[a]=tot;go[tot]=b;val[tot]=c;} //加边inline void bfs(int s,int *dis){memset(vis,0,sizeof vis);dis[s]=0;queue<int>q;q.push(s);vis[s]=1;while(!q.empty()){int u=q.front();q.pop();for(int i=he[u];i;i=ne[i]){int v=go[i];if(!vis[v]){q.push(v);vis[v]=1;dis[v]=dis[u]+val[i]; //计算dis}}}}int main(){cin>>n>>m;for(int i=1;i<=m;i++){int u,v;scanf("%d%d",&u,&v);add(u,v,1);add(v,u,1);}memset(dis1,INF,sizeof dis1);memset(disn,INF,sizeof disn); //开始时全不可达bfs(1,dis1);bfs(n,disn); //bfs找到dis值for(int i=1;i<=n;i++){int ans=min(dis1[n],min(dis1[0]+disn[i],dis1[i]+disn[0])); //具体见图if(ans==INF) cout<<-1<<" "; //如果不可达else cout<<ans<<" ";}return 0;}
完结撒花~