BZOJ 3931: [CQOI2015]网络吞吐量 Dijkstra+最大流
这个没啥难的.
只保留可以转移最短路的边,然后拆点跑一个最大流即可.
#include <bits/stdc++.h> #define N 1004 #define M 250004 #define ll long long #define inf 200000000001 #define setIO(s) freopen(s".in","r",stdin) using namespace std; int s,t,tot,n,m,eds; int vis[N],current[N],id[N][2],hd[N],to[M],nex[M],val[M],cc[N],U[M],V[M],VAL[M]; void add(int u,int v,int c) { nex[++eds]=hd[u],hd[u]=eds,to[eds]=v,val[eds]=c; } struct Dijkstra { ll d[N]; int done[N]; struct Node { int u; long long dis; Node(int u=0,ll dis=0):u(u),dis(dis){} bool operator<(Node b) const { return b.dis<dis; } }; priority_queue<Node>q; void dijkstra(int aa) { memset(d,0x3f,sizeof(d)); for(d[aa]=0,q.push(Node(aa,0));!q.empty();) { Node e=q.top();q.pop(); int u=e.u; if(done[u]) continue; done[u]=1; for(int i=hd[u];i;i=nex[i]) { int v=to[i]; if(d[v]>d[u]+val[i]) { d[v]=d[u]+val[i]; q.push(Node(v, d[v])); } } } } }S,T; int d[M]; struct Edge { ll cap; int u,v; Edge(int u=0,int v=0,ll cap=0):u(u),v(v),cap(cap){} }; queue<int>q; vector<Edge>edges; vector<int>G[N]; void addedge(int u,int v,ll c) { edges.push_back(Edge(u,v,c)); edges.push_back(Edge(v,u,0)); int p=edges.size(); G[u].push_back(p-2); G[v].push_back(p-1); } int BFS() { memset(vis,0,sizeof(vis)); d[s]=0,q.push(s),vis[s]=1; for(;!q.empty();) { int u=q.front(); q.pop(); for(int i=0;i<G[u].size();++i) { Edge e=edges[G[u][i]]; if(e.cap>0&&!vis[e.v]) { vis[e.v]=1; d[e.v]=d[u]+1; q.push(e.v); } } } return vis[t]; } ll dfs(int x,ll cur) { if(x==t) return cur; ll f,flow=0; for(int i=0;i<G[x].size();++i) { Edge e=edges[G[x][i]]; if(e.cap>0 && d[e.v]==d[x]+1) { f=dfs(e.v,min(cur,1ll*e.cap)); if(f) flow+=f,cur-=f,edges[G[x][i]].cap-=f,edges[G[x][i]^1].cap+=f; } if(!cur) break; } return flow; } ll maxflow() { ll flow=0; for(;BFS();) flow+=dfs(s,inf); return flow; } int main() { int i,j,k; // setIO("input"); scanf("%d%d",&n,&m); for(i=1;i<=n;++i) id[i][0]=++tot, id[i][1]=++tot; for(i=1;i<=m;++i) { int a,b,c; scanf("%d%d%d",&a,&b,&c), add(a,b,c),add(b,a,c); U[i]=a,V[i]=b,VAL[i]=c; } for(i=1;i<=n;++i) scanf("%d",&cc[i]); S.dijkstra(1); T.dijkstra(n); addedge(id[1][0],id[1][1],inf); addedge(id[n][0],id[n][1],inf); for(i=2;i<n;++i) { addedge(id[i][0],id[i][1],cc[i]); } for(i=1;i<=m;++i) { int a=U[i],b=V[i]; if(S.d[a]+T.d[b]+VAL[i]==S.d[n]) { addedge(id[a][1],id[b][0],inf); } swap(a,b); if(S.d[a]+T.d[b]+VAL[i]==S.d[n]) { addedge(id[a][1],id[b][0],inf); } } s=id[1][0],t=id[n][1]; printf("%lld\n",maxflow()); return 0; }