题目大意:

在一个有 n×mn\times mn×m 个方格的棋盘中,每个方格中有一个正整数。现要从方格中取数,使任意 2 个数所在方格没有公共边,且取出的数的总和最大。


思路:

这道题明显是一个二分图。我们可以把每个点染色,将i+ji+ji+j为偶数的点连向ttt,否则连向sss。那么对于每一个连sss的白点,将它连向它周围的四个黑点,容量为INFINFINF。跑一边最大流即可。


代码:

#include <cstdio>
#include <iostream>
#include <queue>
#include <cstring>
#define Inf 0x7f
#define INF 1e9
using namespace std;

int n,m,s,t,x,y,k,head[200001],dep[200001],ans,sum,num;

struct edge
{
    int next,to,c;
}e[200001];

void add(int from,int to,int c)
{
    k++;
    e[k].to=to;
    e[k].c=c;
    e[k].next=head[from];
    head[from]=k;
}

bool bfs()
{
    memset(dep,Inf,sizeof(dep));
    dep[s]=0;
    queue<int> q;
    q.push(s);
    while (q.size())
    {
        int u=q.front();
        q.pop();
        for (int i=head[u];i;i=e[i].next)
        {
            int v=e[i].to;
            if (dep[v]>dep[u]+1&&e[i].c)
            {
                dep[v]=dep[u]+1;
                q.push(v);
            }
        }
    }
    return dep[t]<Inf;
}

int dfs(int u,int low)
{
    int lows=0;
    if (u==t) return low;
    for (int i=head[u];i;i=e[i].next)
    {
        int v=e[i].to;
        if (dep[v]==dep[u]+1&&e[i].c)
        {
            lows=dfs(v,min(e[i].c,low));
            if (lows)
            {
                e[i].c-=lows;
                e[i^1].c+=lows;
                return lows;
            }
        }
    }
    return 0;
}

  //以上为最大流模板

int main()
{
    scanf("%d%d",&n,&m);
    k=1;
    s=n*m+4;
    t=n*m+5;
    for (int i=1;i<=n;i++)
     for (int j=1;j<=m;j++)
     {
     	scanf("%d",&x);
     	num+=x;
     	y=i*m-m+j;
     	if (!((i+j)%2))  //黑点
     	{
     		add(y,t,x);
     		add(t,y,0);
     	}
     	else  //白点
     	{
     		add(s,y,x);
     		add(y,s,0);
     		if (i>1)  //连上面的黑点
     		{
     			add(y,y-m,INF);
     			add(y-m,y,0);
     		}
     		if (i<n)  //连下面的黑点
     		{
     			add(y,y+m,INF);
     			add(y+m,y,0);
     		}
     		if (j>1)  //连左边的黑点
     		{
     			add(y,y-1,INF);
     			add(y-1,y,0);
     		}
     		if (j<m)  //连右边的黑点
     		{
     			add(y,y+1,INF);
     			add(y+1,y,0);
     		}
     	}
     }
     
    while (bfs())
    {
        while (sum=dfs(s,INF+1))
         ans+=sum;
    }
    printf("%d\n",num-ans);
    return 0;
}