还是最大权闭合子图的题
对于每个中转站$k$:$link(k,T,P_k)$
对于每个用户$i$、中转站$A_i,B_i$、贡献$C_i$
$link(S,i,C_i)$
$link(i,A_i,inf)$
$link(i,B_i,inf)$
蓝后就可以跑最小割辣
#include<iostream> #include<cstdio> #include<cstring> #include<queue> using namespace std; #define N 100005 #define W 4000005 #define inf 2000000000 int n,m,d[N],cur[N],tot,S,T; bool vis[N]; queue <int> h; int cnt=1,hd[N],nxt[W],ed[N],poi[W],val[W]; inline void adde(int x,int y,int v){ nxt[ed[x]]=++cnt, hd[x]=hd[x]?hd[x]:cnt, ed[x]=cnt, poi[cnt]=y, val[cnt]=v; } inline void link(int x,int y,int v){adde(x,y,v),adde(y,x,0);} bool bfs(){ memset(vis,0,sizeof(vis)); h.push(S); vis[S]=1; while(!h.empty()){ int x=h.front(); h.pop(); for(int i=hd[x];i;i=nxt[i]){ int to=poi[i]; if(!vis[to]&&val[i]>0) vis[to]=1,d[to]=d[x]+1,h.push(to); } }return vis[T]; } int dfs(int x,int a){ if(x==T||a==0) return a; int F=0,f; for(int &i=cur[x];i;i=nxt[i]){ int to=poi[i]; if(d[to]==d[x]+1&&(f=dfs(to,min(a,val[i])))>0) a-=f,F+=f,val[i]-=f,val[i^1]+=f; if(!a) break; }return F; } int dinic(){ int re=0; while(bfs()){ for(int i=1;i<=T;++i) cur[i]=hd[i]; re+=dfs(S,inf); }return re; } int main(){ scanf("%d%d",&n,&m); int q1,q2,w; S=n+m+1; T=S+1; for(int i=1;i<=n;++i) scanf("%d",&w),link(i,T,w); for(int i=1;i<=m;++i){ scanf("%d%d%d",&q1,&q2,&w); tot+=w; link(S,i+n,w); link(i+n,q1,inf); link(i+n,q2,inf); }printf("%d",tot-dinic()); return 0; }