题意:
求一个带权有向图的1到n的最短路径数量,两点间会有重边,两条最短路径只有在存在一条边及以上不同时认为不同(所以重边算一条边),若城市n无法到达则只输出一个(‘No answer’)。
题解:
和P1144差不多,但是P1144是无权图,所以可以直接bfs,本题是带权图,就需要使用dijkstra算法求最短路了,更新数量时,如果$dis[v]$需要松弛,则说明存在一条新的边缩短了$1$到$v$的最短距离,则最短路径数就是$dis[u]$,如果不需要松弛,就是$cnt[v]+=cnt[u]$,不存在$dis[v]<dis[u]$的情况。因为重边算一条边,所以在存图时需要去重,先用矩阵存,再转成邻接表或者直接使用矩阵存图跑$dijkstra$算法即可。
AC代码:
#include <iostream> #include <queue> #include <algorithm> #include <vector> #include <cstring> using namespace std; int dis[2005],vis[2005],cnt[2005]; struct edge { int to,w; edge(int a,int b) { to=a,w=b; } bool operator<(const edge &a)const { return this->w>a.w; } }; vector<edge>node[2005]; void dijkstra(int n) { memset(dis,0x3f3f3f3f,sizeof(dis)); memset(vis,0,sizeof(vis)); memset(cnt,0,sizeof(cnt)); dis[1]=0,cnt[1]=1; priority_queue<edge>que; que.push(edge(1,dis[1])); while(!que.empty()) { edge u=que.top(); que.pop(); if(vis[u.to]) continue; vis[u.to]=1; for(int i=0;i<node[u.to].size();++i) { edge v=node[u.to][i]; if(!vis[v.to]) { if(dis[v.to]==dis[u.to]+v.w) cnt[v.to]+=cnt[u.to]; else if(dis[v.to]>dis[u.to]+v.w) { dis[v.to]=dis[u.to]+v.w; cnt[v.to]=cnt[u.to]; que.push(edge(v.to,dis[v.to])); } } } } /*for(int i=1;i<=n;++i) cout<<dis[i]<<" "; cout<<endl;*/ if(dis[n]>=0x3f3f3f3f) cout<<"No answer"<<endl; else cout<<dis[n]<<" "<<cnt[n]<<endl; } int graph[2005][2005]; int main() { memset(graph,0x3f3f3f3f,sizeof(graph)); int n,m,a,b,c; cin>>n>>m; for(int i=0;i<m;++i) { cin>>a>>b>>c; if(graph[a][b]>c) graph[a][b]=c; } for(int i=1;i<=n;++i) for(int j=1;j<=n;++j) if(graph[i][j]<0x3f3f3f3f) node[i].push_back(edge(j,graph[i][j])); dijkstra(n); return 0; }