【题目名称】:方格取数问题
Description【问题描述】:
在一个有 m*n 个方格的棋盘中(1<=m,n<=30),每个方格中有一个正整数。现要从方格中取数,使任意2个数所在方格没有公共边,且取出的数的总和最大。试设计一个满足要求的取数算法。 【编程任务】: 对于给定的方格棋盘,按照取数要求编程找出总和最大的数。 Input由文件input.txt提供输入数据。文件第1 行有2 个正整数 m和 n,分别表示棋盘的行数和列数。接下来的m行,每行有n个正整数,表示棋盘方格中的数。
Output
程序运行结束时,将取数的最大总和输出到文件output.txt中。
Sample Input
3 3 1 2 3 3 2 3 2 3 1 Sample Output 11 【题目分析】: 【问题分析】二分图点权最大独立集,转化为最小割模型,从而用最大流解决。
【建模方法】
首先把棋盘黑白染色,使相邻格子颜色不同,所有黑色格子看做二分图X集合中顶点,白色格子看做Y集合顶点,建立附加源S汇T。
1、从S向X集合中每个顶点连接一条容量为格子中数值的有向边。
2、从Y集合中每个顶点向T连接一条容量为格子中数值的有向边。 3、相邻黑白格子Xi,Yj之间从Xi向Yj连接一条容量为无穷大的有向边。求出网络最大流,要求的结果就是所有格子中数值之和减去最大流量。
【建模分析】
这是一个二分图最大点权独立集问题,就是找出图中一些点,使得这些点之间没有边相连,这些点的权值之和最大。独立集与覆盖集是互补的,求最大点权独立集可以转化为求最小点权覆盖集(最小点权支配集)。最小点权覆盖集问题可以转化为最小割问题解决。结论:最大点权独立集 = 所有点权 - 最小点权覆盖集 = 所有点权 - 最小割集 = 所有点权 - 网络最大流。
对于一个网络,除去冗余点(不存在一条ST路径经过的点),每个顶点都在一个从S到T的路径上。割的性质就是不存在从S到T的路径,简单割可以认为割边关联的非ST节点为割点,而在二分图网络流模型中每个点必关联到一个割点(否则一定还有增广路,当前割不成立),所以一个割集对应了一个覆盖集(支配集)。最小点权覆盖集就是最小简单割,求最小简单割的建模方法就是把XY集合之间的变容量设为无穷大,此时的最小割就是最小简单割了。
有关二分图最大点权独立集问题,更多讨论见《最小割模型在信息学竞赛中的应用》作者胡伯涛。
【C++源码】: #include<iostream> using namespace std;const int maxn=1024;
const int oo=0x7FFFFFFF/2-5; const int dx[]={-1,0,1,0}, dy[]={0,-1,0,1};struct NODE{
int st,ed,f; NODE *op,*next; NODE(){op=next=NULL;} }*g[maxn],*now[maxn],*pre[maxn];typedef int Arr1[maxn];
int m,n;
int ST,ED; int sum=0; int a[55][55]={0};Arr1 dis={0};
Arr1 back={0}; Arr1 sumd={0};bool flag;
bool cor[55][55]={0};void add(int st,int ed,int f)
{ NODE *x,*y; x=new NODE; y=new NODE; x->op=y; y->op=x; x->st=st,x->ed=ed,x->f=f,x->next=g[st],g[st]=x; y->st=ed,y->ed=st,y->f=0,y->next=g[ed],g[ed]=y; }void Color(){
int i,j,k; for( i=1 ;i<=m ;i+=2 ) cor[1][i]=1; for( i=2 ;i<=n ;i++ ) { for( j=1 ;j<=m ;j++ ) cor[i][j]=!cor[i-1][j]; } }void init(){
int st,ed,v; int i,j,k,t; scanf("%d%d",&n,&m); Color(); ST=n*m+1; ED=n*m+2; for( i=1 ;i<=n*m+10 ;i++ ) g[i]=NULL; for( i=1 ;i<=n ;i++ ) for( j=1 ;j<=m ;j++ ) { scanf("%d",&a[i][j]);//cin>>a[i][j]; sum+=a[i][j]; } for( i=1 ;i<=n ;i++ ) for( j=1 ;j<=m ;j++ ) { int num=m*(i-1)+j; if( cor[i][j] ) add(ST,num,a[i][j]); else add(num,ED,a[i][j]); if( cor[i][j] ) for( k=0 ;k<4 ;k++ ) { int nx,ny; nx=i+dx[k]; ny=j+dy[k]; if( nx<1 || nx>n || ny<1 || ny>m ) continue; int num2=m*(nx-1)+ny; if( cor[i][j]!=cor[nx][ny] ) add(num,num2,oo); } } n=n*m+2; for( int i=1 ;i<=n ;i++ ) now[i]=g[i]; }void SAP(){
int i,j,k; int ans=0; int flow=0x7FFFFFFF; for( sumd[0]=n,i=ST ;dis[ST]<n ; ) { flag=false; back[i]=flow; for( NODE *t=now[i] ;t!=NULL ;t=t->next ) { j=t->ed; if( t->f<=0 || dis[i]!=dis[j]+1 ) continue; flag=true; now[i]=t; pre[j]=t; flow<?=t->f; i=j; if( i==ED ) { ans+=flow; while( i!=ST ) { pre[i]->f-=flow; pre[i]->op->f+=flow; i=pre[i]->st; } flow=0x7FFFFFFF; } break; } if( flag ) continue; int Min=n-1; for( NODE *t=g[i] ;t!=NULL ;t=t->next ) if( t->f>0 && dis[t->ed]<Min ) { now[i]=t; Min=dis[t->ed]; } if( !(--sumd[dis[i]]) ) break; sumd[dis[i]=Min+1]++; if( i!=ST ) i=pre[i]->st,flow=back[i]; } cout<<sum-ans<<endl; }int main()
{ init(); SAP(); return 0; }