[SDOI2009]
- BZOJ题面谁写的,出来挨打
- 费用流垃圾套路题,拆点即可
#include<bits/stdc++.h>
#define rep(i,a,b) for(ll i=(a);i<=(b);i++)
#define per(i,a,b) for(ll i=(a);i>=(b);i--)
#define st(x) x
#define ed(x) x+n
#define ll long long
using namespace std;
const ll inf=1e9;
const ll N=1e6;
struct node{ll y,v,w,n;}e[N];
ll lin[N],flow[N/100],pre[N/100],v[N/100],d[N/100],S,T,len=1,x,y,w,n,m;
ll maxflow=0,ans=0;
void read(ll x,ll y,ll v,ll w)
{e[++len].y=y,e[len].v=v,e[len].w=w,e[len].n=lin[x],lin[x]=len;}
void add(ll x,ll y,ll v,ll w)
{read(x,y,v,w),read(y,x,0,-w);}
bool SPFA(){
queue<ll> q;
memset(v,0,sizeof(v));
memset(d,0x3f,sizeof(d));
q.push(S),v[S]=1,d[S]=0,flow[S]=inf;
while(q.size()){
ll x=q.front();q.pop();v[x]=0;
for(ll i=lin[x];i;i=e[i].n){
ll y=e[i].y;
if(!e[i].v)continue;
if(d[y]>d[x]+e[i].w){
pre[y]=i;
d[y]=d[x]+e[i].w;
flow[y]=min(e[i].v,flow[x]);
if(!v[y])v[y]=1,q.push(y);
}
}
}return d[T]<inf;
}
void upd(){
ll x=T;
while(x!=S){
ll i=pre[x];
e[i].v-=flow[T];
e[i^1].v+=flow[T];
x=e[i^1].y;
}
maxflow+=flow[T];
ans+=flow[T]*d[T];
}
int main()
{
scanf("%lld%lld",&n,&m); S=ed(1),T=st(n);
rep(i,1,n)add(st(i),ed(i),1,0);
rep(i,1,m){
scanf("%lld%lld%lld",&x,&y,&w);
if(x==1&&y==n)add(ed(x),st(y),1,w);
else add(ed(x),st(y),inf,w);
}
while(SPFA()) upd();
cout<<maxflow<<" "<<ans;
return 0;
}