题意:
给出一个n个结点m条边有向图,可能有负权边;
但是存在负权边a->b则不会有某个路径可以从b到a;
求一个源点s到所有点的最短路(无解输出"NO PATH");
n<=25000,m<=150000;
题解:
高高兴兴的写了一发spfa,O(km)嘛;
然后就TLE了,这题丧心病狂的把spfa卡掉了;
这时候理所当然的想到了dij+heap,写到一半想起来不支持负权边;
所以这个不是一个简单的单源最短路问题;
题中有一个重要条件就是负权边不会回去;
我本以为只是排除了负权环的存在,但是事实上,这个性质使负权边不可能在强连通分量内;
那么每个强连通分量内的最短路可以用dij实现;
强连通分量缩点后是DAG,直接DP传一下进入强连通分量的最短路就可以了;
时间复杂度O(nlogn);
HINT:
每次dij时是没有源点的,直接将所有点加入heap做最短路;
不可到达的强连通分量不能将最短路下传,防止在最后判断是否为inf时错误;
USACO数据还是naive;
为了写这题还去学了dij+heap然后过不了也是醉;
最近写的代码越来越长是错觉吗。。。
代码:
#include<queue> #include<stack> #include<vector> #include<stdio.h> #include<string.h> #include<algorithm> #define N 25500 #define pr pair<int,int> using namespace std; vector<int>to[N],val[N],son[N]; priority_queue<pr,vector<pr >,greater<pr > >poq; queue<int>q; stack<int>st; int f[N],dis[N]; int deep[N],low[N],belong[N],in[N],cnt,tot; bool ins[N],cov[N],vis[N]; void tarjan(int x) { deep[x]=low[x]=++cnt; st.push(x),ins[x]=1; int i,y; for(i=0;i<to[x].size();i++) { if(!deep[y=to[x][i]]) tarjan(y),low[x]=min(low[x],low[y]); else if(ins[y]) low[x]=min(low[x],deep[y]); } if(deep[x]==low[x]) { tot++; int k; do { k=st.top(),st.pop(); ins[k]=0; belong[k]=tot; son[tot].push_back(k); }while(k!=x); } } void dij(int x) { int i,j,k,y; for(i=0;i<son[x].size();i++) poq.push(pr(dis[son[x][i]],son[x][i])); while(!poq.empty()) { k=poq.top().second; poq.pop(); if(vis[k]) continue; vis[k]=1; for(i=0;i<to[k].size();i++) { if(belong[y=to[k][i]]==x) if(dis[y]>dis[k]+val[k][i]) { dis[y]=dis[k]+val[k][i]; poq.push(pr(dis[y],y)); } } } } int main() { int n,m1,m2,s,i,j,k,x,y,v; scanf("%d%d%d%d",&n,&m1,&m2,&s); for(i=1;i<=m1;i++) { scanf("%d%d%d",&x,&y,&v); to[x].push_back(y); val[x].push_back(v); to[y].push_back(x); val[y].push_back(v); } for(i=1;i<=m2;i++) { scanf("%d%d%d",&x,&y,&v); to[x].push_back(y); val[x].push_back(v); } for(i=1;i<=n;i++) if(!deep[i]) tarjan(i); for(x=1;x<=n;x++) { for(i=0;i<to[x].size();i++) { if(belong[y=to[x][i]]!=belong[x]) { in[belong[y]]++; } } } for(i=1;i<=tot;i++) if(!in[i]) q.push(i); memset(dis,0x3f,sizeof(dis)); dis[s]=0,cov[belong[s]]=1; while(!q.empty()) { x=q.front(),q.pop(); if(cov[x]) dij(x); for(j=0;j<son[x].size();j++) { k=son[x][j]; for(i=0;i<to[k].size();i++) { if(belong[y=to[k][i]]!=x) { if(cov[x]) { cov[belong[y]]=1; dis[y]=min(dis[k]+val[k][i],dis[y]); } in[belong[y]]--; if(!in[belong[y]]) q.push(belong[y]); } } } } for(i=1;i<=n;i++) { if(dis[i]==0x3f3f3f3f) puts("NO PATH"); else printf("%d\n",dis[i]); } return 0; }
USACO 2011 Jan Gold 3. Roads and Planes
原文地址:http://blog.csdn.net/ww140142/article/details/46494505