【网络流24题】方格取数问题 题解

题目传送门

题目大意: 有一个 n × m n\times m n×m 的网格图,每个格子上有一个数字,现在要求取出数字的和尽可能大,并且取出的数字两两不相邻。

题解

直接跑最大流好像不太可行,考虑转化一下。

有一个很显然的柿子:取的+没取的=所有数字之和,现在要求取的尽可能大,也就是要让没取的尽可能小。

那么考虑将网格图黑白染色,源点连向白点,黑点连向汇点,流量为点上的数,然后白点向相邻的黑点连边,流量无限,这样的话这些边不会被割掉,只有和源汇相连的边才会被割掉,而那些边被割掉就代表不取他们。

跑最大流时,要使得一个点与源汇之间的边不被割掉,只有将和他相邻的点都割掉,这样就满足了不会取到相邻的格子,由于最大流等于最小割,所以最大流就是最小的没取的数之和

代码如下:

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define maxn 100010
#define inf 999999999

int n,m,S,T;
int a[110][110],ans=0;
struct edge{int y,z,next;};
edge e[maxn<<1];
int first[maxn],len=1;
void buildroad(int x,int y,int z)
{
	e[++len]=(edge){y,z,first[x]};
	first[x]=len;
}
int f1[4]={0,-1,0,1},f2[4]={-1,0,1,0};
int h[maxn],q[maxn],st,ed;
bool bfs()
{
	memset(h,0,sizeof(h));
	st=ed=1;q[st]=S;h[S]=1;
	while(st<=ed)
	{
		int x=q[st++];
		for(int i=first[x];i;i=e[i].next)
		if(!h[e[i].y]&&e[i].z)h[q[++ed]=e[i].y]=h[x]+1;
	}
	return h[T];
}
int dfs(int x,int flow)
{
	if(x==T)return flow;
	int tt=0,p;
	for(int i=first[x];i;i=e[i].next)
	{
		int y=e[i].y;
		if(h[y]==h[x]+1&&e[i].z)
		{
			p=dfs(y,min(e[i].z,flow-tt));tt+=p;
			e[i].z-=p;e[i^1].z+=p;
			if(tt==flow)break;
		}
	}
	if(!tt)h[x]=0;
	return tt;
}

int main()
{
	scanf("%d %d",&n,&m);S=n*m+1,T=S+1;
	for(int i=1;i<=n;i++)
	for(int j=1;j<=m;j++)
	{
		scanf("%d",&a[i][j]);ans+=a[i][j];
		if(!((i%2)^(j%2)))buildroad(S,(i-1)*m+j,a[i][j]),buildroad((i-1)*m+j,S,0);
		else buildroad((i-1)*m+j,T,a[i][j]),buildroad(T,(i-1)*m+j,0);
		if(!((i%2)^(j%2)))for(int k=0;k<4;k++)
		{
			int x=i+f1[k],y=j+f2[k];
			if(x<1||x>n||y<1||y>m)continue;
			buildroad((i-1)*m+j,(x-1)*m+y,inf),buildroad((x-1)*m+y,(i-1)*m+j,0);
		}
	}
	while(bfs())ans-=dfs(S,inf);
	printf("%d",ans);
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值