传送门
日常费用流水题。
直接按照题意模拟求一下最小费用最大流就行了。
注意是点不能重合因此需要拆点。
代码:
#include<bits/stdc++.h>
#define N 505
#define M 20005
using namespace std;
struct edge{int v,next,c,w;};
int n,m;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
struct MCMF{
edge e[M<<2];
int first[N],cnt,s,t,d[N],pred[N],pos[N],flow[N];
bool in[N];
inline void init(){memset(first,-1,sizeof(first)),cnt=-1,s=1+n,t=n;}
inline void addedge(int u,int v,int c,int w){e[++cnt].v=v,e[cnt].c=c,e[cnt].w=w,e[cnt].next=first[u],first[u]=cnt;}
inline void add(int u,int v,int c,int w){addedge(u,v,c,w),addedge(v,u,0,-w);}
inline bool spfa(){
queue<int>q;
for(int i=1;i<=n*2;++i)d[i]=0x3f3f3f3f;
memset(in,false,sizeof(in)),d[s]=0,in[s]=1,pred[t]=-1,flow[s]=0x3f3f3f3f,q.push(s);
while(!q.empty()){
int x=q.front();
q.pop(),in[x]=0;
for(int i=first[x];~i;i=e[i].next){
int v=e[i].v;
if(!e[i].c||d[v]<=d[x]+e[i].w)continue;
d[v]=d[x]+e[i].w,pred[v]=x,pos[v]=i,flow[v]=min(flow[x],e[i].c);
if(!in[v])in[v]=1,q.push(v);
}
}
return d[t]!=0x3f3f3f3f;
}
inline pair<int,int> solve(){
int ret1=0,ret2=0;
for(int w=t;spfa();w=t){
ret1+=flow[t]*d[t],ret2+=flow[t];
while(w!=s)e[pos[w]].c-=flow[t],e[pos[w]^1].c+=flow[t],w=pred[w];
}
return make_pair(ret1,ret2);
}
}mcmf;
int main(){
n=read(),m=read(),mcmf.init();
for(int i=2;i<n;++i)mcmf.add(i,i+n,1,0);
for(int i=1,u,v,w;i<=m;++i)u=read(),v=read(),w=read(),mcmf.add(u+n,v,1,w);
pair<int,int>t=mcmf.solve();
printf("%d %d",t.second,t.first);
return 0;
}