码迷,mamicode.com
首页 > 其他好文 > 详细

bzoj3876: [Ahoi2014]支线剧情 费用流

时间:2016-06-18 15:29:14      阅读:287      评论:0      收藏:0      [点我收藏+]

标签:

按照有下界网络流求可行流的方法求出最小费用可行流。

写的费用流多路增广,比裸ek快一点。

其实这题有很优越的建图方法,然而并不会。

#include<bits/stdc++.h>
#define N 309
using namespace std;
int cost,n;
int d[N],s=N-1,t=N-2;
bool p[N];
const int inf=1061109567;
struct edge{
	edge* s;
	int v,w,cp;
}e[N*N],*back(e),*h[N];
void add(
int u,int v,int w,int cp){
	h[u]=&(*back++
	=(edge){h[u],v,w,cp});
	h[v]=&(*back++
	=(edge){h[v],u,-w,0});
}
bool spfa(){
	static bool a[N];
	queue<int> q;
	q.push(s);
	memset(d,63,sizeof d);
	d[s]=0;
	while(q.size()){
		int u=q.front();
		q.pop();
		a[u]=0;
		for(edge* i=h[u];i;i=i->s)
			if(i->cp
			&&d[i->v]>d[u]+i->w){
				d[i->v]=d[u]+i->w;
				if(!a[i->v]){
					a[i->v]=1;
					q.push(i->v);
				}
			}
	}
	return d[t]!=inf;
}
int dfs(int u,int a){
	int s=0;
	p[u]=1;
	for(edge* i=h[u];s!=a
	&&i;i=i->s){
		if(!i->cp||p[i->v]
		||d[i->v]!=d[u]+i->w)
			continue;
		int j=min(a-s,i->cp);
		if(i->v==t
		||(j=dfs(i->v,j))){
			s+=j;
			cost+=i->w*j;
			i->cp-=j;
			e[i-e^1].cp+=j;
		}
	}
	return s;
}
int main(){
	scanf("%d",&n);
	for(int i=1;i<=n;++i){
		int j,v,w;
		scanf("%d",&j);
		add(i,t,0,j);
		add(i,1,0,inf);
		while(j--){
			scanf("%d%d",&v,&w);
			add(s,v,w,1);
			add(i,v,w,inf);
		}
	}
	while(spfa()){
		memset(p,0,sizeof p);
		dfs(s,inf);
	}
	printf("%d\n",cost);
}

  

bzoj3876: [Ahoi2014]支线剧情 费用流

标签:

原文地址:http://www.cnblogs.com/f321dd/p/5596196.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!