bzoj4519 不同的最小割 分治&最小割

       同zjoi的那倒最小割,跑最小割之后按S和T两个集合递归分治。

AC代码如下:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define N 1005
#define M 20005
using namespace std;

int n,m,tot=1,sta,gol,cnt,fst[N],cur[N],pnt[M],len[M],nxt[M],d[N],h[N],num[N],fa[N],a[N],b[N],c[N];
bool vis[N];
void add(int x,int y,int z){
	pnt[++tot]=y; len[tot]=z; nxt[tot]=fst[x]; fst[x]=tot;
}
void bfs(){
	memset(num,0,sizeof(num));
	memset(d,0x3f,sizeof(d)); d[gol]=0;
	int head=0,tail=1; h[1]=gol;
	while (head<tail){
		int x=h[++head],p; num[d[x]]++;
		for (p=fst[x]; p; p=nxt[p]) if (len[p^1]){
			int y=pnt[p];
			if (d[x]+1<d[y]){
				d[y]=d[x]+1; h[++tail]=y;
			}
		}
	}
}
int up(){
	int i,tmp=1000000000;
	for (i=gol; i!=sta; i=pnt[fa[i]^1]) tmp=min(tmp,len[fa[i]]);
	for (i=gol; i!=sta; i=pnt[fa[i]^1]){
		len[fa[i]]-=tmp; len[fa[i]^1]+=tmp;
	}
	return tmp;
}
int isap(){
	int i;
	for (i=2; i<=tot; i++){ len[i]=len[i^1]=(len[i]+len[i^1])>>1; }
	bfs(); memcpy(cur,fst,sizeof(fst));
	int x=sta,flow=0,p; bool flag;
	while (d[x]<n){
		if (x==gol){ flow+=up(); x=sta; }
		flag=1;
		for (p=cur[x]; p; p=nxt[p]) if (len[p]){
			int y=pnt[p];
			if (d[y]+1==d[x]){
				fa[y]=cur[x]=p; x=y;
				flag=0; break;
			}
		}
		if (flag){
			int mn=n-1; cur[x]=fst[x];
			for (p=fst[x]; p; p=nxt[p])
				if (len[p]) mn=min(mn,d[pnt[p]]);
			num[d[x]]--; if (!num[d[x]]) break;
			d[x]=mn+1; num[d[x]]++;
			if (x!=sta) x=pnt[fa[x]^1];
		}
	}
	return flow;
}
void dfs(int x){
	vis[x]=1; int p;
	for (p=fst[x]; p; p=nxt[p]) if (len[p]){
		int y=pnt[p]; if (!vis[y]) dfs(y);
	}
}
void solve(int l,int r){
	if (l==r) return;
	sta=a[l]; gol=a[r]; int flow=isap(),i;
	c[++cnt]=flow;
	memset(vis,0,sizeof(vis)); dfs(sta);
	int x=l,y=r;
	for (i=l; i<=r; i++)
		if (vis[a[i]]) b[x++]=a[i]; else b[y--]=a[i];
	for (i=l; i<=r; i++) a[i]=b[i];
	if (l<x) solve(l,x-1); if (y<r) solve(y+1,r);
}
int main(){
	scanf("%d%d",&n,&m); int i,x,y,z;
	for (i=1; i<=m; i++){
		scanf("%d%d%d",&x,&y,&z);
		add(x,y,z); add(y,x,z);
	}
	for (i=1; i<=n; i++) a[i]=i; solve(1,n);
	sort(c+1,c+cnt+1); int ans=1;
	for (i=2; i<=cnt; i++)
		if (c[i]!=c[i-1]) ans++;
	printf("%d\n",ans);
	return 0;
}


by lych

2016.4.22

你可能感兴趣的:(分治,网络流,最大流,isap)