洛谷P2057 【SHOI2007】善意的投票

洛谷P2057 【SHOI2007】善意的投票

题目链接

这道题是最小割的一个经典应用:划分集合。

题目的意思就是就是将所有的小朋友分为两个集合:同意睡觉和不同意睡觉的。不同的集合之间的边都要断开。

我们设\(S\)为投票结果为不想睡觉的小朋友(颜色为0)的集合;\(T\)为投票结果为想睡觉的小朋友(颜色为1)的集合。然后对于一个小朋友\(i\),设他的“颜色”为x,那么我们就连两条边\((S,i,[x!=0]),(i,T,[x!=1])\)。第一条边表示该小朋友属于\(S\)集合,第二条边表示该小朋友属于\(T\)集合。

因为投与自己意愿相反的票会产生冲突,所以需要给定流量。

然后对于一对好朋友\(i,j\),我们连\((i,j,1)\)的双向边。

实际操作中,流量为0的边自然可以不连。

答案就是最小割。这是因为,如果\(S\)\(T\)之间还有流量,说明还有至少一对有冲突的好朋友存在。从这个角度来想,那么答案和最小割等价的。

如果要问最后小朋友们投的是那些票,那就看最小割割的是哪些边。如果割的是\((i,j)\),表示保留冲突。如果割的是\((s,i)\)\((i,T)\),表示\(i\)投了意愿相反的票。

代码:

#include<bits/stdc++.h>
#define ll long long
#define N 305

using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;}

int n,m;
int S,T;
struct load {
	int to,next;
	int flow;
}s[N*N<<2];
int h[N],cnt=1;
void add(int i,int j,int flow) {
	s[++cnt]=(load) {j,h[i],flow};h[i]=cnt;
	s[++cnt]=(load) {i,h[j],0};h[j]=cnt;
}
int dis[N],gap[N];
int dfs(int v,int maxf) {
	if(v==T) return maxf;
	int ret=0;
	for(int i=h[v];i;i=s[i].next) {
		int to=s[i].to;
		if(s[i].flow&&dis[to]+1==dis[v]) {
			int dlt=dfs(to,min(maxf-ret,s[i].flow));
			s[i].flow-=dlt;
			s[i^1].flow+=dlt;
			ret+=dlt;
			if(ret==maxf||dis[S]>=n+2) return ret;
		}
	}
	if(!(--gap[dis[v]])) dis[S]=n+2;
	gap[++dis[v]]++;
	return ret;
}
int sap() {
	memset(gap,0,sizeof(gap));
	memset(dis,0,sizeof(dis));
	gap[0]=n+2;
	int ans=0;
	while(dis[S]<n+2) ans+=dfs(S,1<<29);
	return ans;
}
void Init() {
	cnt=1;
	memset(h,0,sizeof(h));
}

int main() {
	n=Get(),m=Get(); 
	Init();
	T=n+1;
	for(int i=1;i<=n;i++) {
		int a=Get();
		if(a==1) add(S,i,1);
		else add(i,T,1);
	}
	for(int i=1;i<=m;i++) {
		int a=Get(),b=Get();
		add(a,b,1),add(b,a,1);
	}
	cout<<sap()<<"\n";
	return 0;
}

posted @ 2018-12-05 16:01  hec0411  阅读(136)  评论(0编辑  收藏  举报