zoukankan      html  css  js  c++  java
  • 【解题报告】BSOI2550 方格取数游戏 (转)

    【题目名称】:方格取数问题
    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;
    }

  • 相关阅读:
    【转】利用MVC模式开发Java应用程序[组图]
    [转]JAVA三大框架SSH和MVC
    二进制转十进制
    MPlayerX For Mac白屏问题
    输入法切换设置
    【学习笔记】【C语言】选择结构-switch
    【学习笔记】【C语言】选择结构-if
    【学习笔记】【C语言】流程控制
    Mac OS X中开启或关闭显示隐藏文件
    Xcode6 模拟器不显示键盘
  • 原文地址:https://www.cnblogs.com/Open_Source/p/1904964.html
Copyright © 2011-2022 走看看