【bzoj 2132】圈地计划(最小割)

45 篇文章 0 订阅
16 篇文章 0 订阅

Time Limit: 2 Sec Memory Limit: 256 MB
Submit: 923 Solved: 418
[Submit][Status][Discuss]

Description

最近房地产商GDOI(Group of Dumbbells Or Idiots)从NOI(Nuts Old Idiots)手中得到了一块开发土地。据了解,这块土地是一块矩形的区域,可以纵横划分为N×M块小区域。GDOI要求将这些区域分为商业区和工业区来开发。根据不同的地形环境,每块小区域建造商业区和工业区能取得不同的经济价值。更具体点,对于第i行第j列的区域,建造商业区将得到Aij收益,建造工业区将得到Bij收益。另外不同的区域连在一起可以得到额外的收益,即如果区域(I,j)相邻(相邻是指两个格子有公共边)有K块(显然K不超过4)类型不同于(I,j)的区域,则这块区域能增加k×Cij收益。经过Tiger.S教授的勘察,收益矩阵A,B,C都已经知道了。你能帮GDOI求出一个收益最大的方案么?

Input

输入第一行为两个整数,分别为正整数N和M,分别表示区域的行数和列数;第2到N+1列,每行M个整数,表示商业区收益矩阵A;第N+2到2N+1列,每行M个整数,表示工业区收益矩阵B;第2N+2到3N+1行,每行M个整数,表示相邻额外收益矩阵C。第一行,两个整数,分别是n和m(1≤n,m≤100);

任何数字不超过1000”的限制

Output

输出只有一行,包含一个整数,为最大收益值。

Sample Input

3 3
1 2 3
4 5 6
7 8 9
9 8 7
6 5 4
3 2 1
1 1 1
1 3 1
1 1 1

Sample Output

81
对于100%的数据有N,M≤100

**【题解】【网络流,最小割(=最大流)】
【黑白染色,黑点选工业区与源点连边,选商业区与汇点连边,流量为收益;白点选商业区与源点连边,选工业区与汇点连边,流量为收益; 相邻两点之间连边(因为染色本就将它们染成不同的颜色),流量为两点选不同颜色的收益和。别忘了将所有边的流量加起来,最后减去最小割,就是答案】 **

#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int a[1000010],next[1000010],p[200010],remain[1000010],tot;
int dis[200010],cur[200010];
int val[110][110];
int n,m,S,T;
long long mincut,sum,s,ans;

inline void add(int x,int y,int flow)
{
    tot++; a[tot]=y; next[tot]=p[x]; p[x]=tot; remain[tot]=flow;
    tot++; a[tot]=x; next[tot]=p[y]; p[y]=tot; remain[tot]=0;
}

inline bool bfs(int s,int t)
{
    queue<int>que;
    memset(dis,-1,sizeof(dis));
    for (int i=s;i<=t;++i) cur[i]=p[i];
    que.push(s); dis[s]=0;
    while (!que.empty())
     {
        int u,v;
        u=que.front(); que.pop();
        v=p[u];
        while (v!=-1)
         {
            if (remain[v]&&dis[a[v]]<0)
             {
                dis[a[v]]=dis[u]+1;
                que.push(a[v]);
               }
            v=next[v];
          }
     }
    if (dis[t]<0) return false;
     else return true;
}
inline int dfs(int now,int t,int flow)
{
    if (now==t||!flow) return flow;
    int u=cur[now],s,s1=0;
    while (u!=-1)
     {
        cur[now]=u;
        if (dis[a[u]]==dis[now]+1&&(s=dfs(a[u],t,min(flow,remain[u]))))
         {
            s1+=s; flow-=s;
            remain[u]-=s; remain[u^1]+=s;
            if (!flow) break;
          }
        u=next[u];
     }
    return s1;
}
int main()
{
    int i,j,k;
    memset(p,-1,sizeof(p));
    memset(next,-1,sizeof(next));
    scanf("%d%d",&n,&m);
    T=n*m+1; tot=-1;
    for (i=1;i<=n;++i)
     for (j=1;j<=m;++j)
       {
        int x,k;
        k=(i-1)*m+j;
        scanf("%d",&x);
        if((i+j)&1) add(S,k,x);
         else add(k,T,x); 
        sum+=x;
        }  
    for (i=1;i<=n;++i)
     for (j=1;j<=m;++j)
      {
        int x,k=(i-1)*m+j;
        scanf("%d",&x);
        if (!((i+j)&1)) add(S,k,x);
         else add(k,T,x);
        sum+=x;
      }
    for (i=1;i<=n;++i)
     for (j=1;j<=m;++j)
      scanf("%d",&val[i][j]);
    for (i=1;i<=n;++i)
     for (j=1;j<=m;++j)
      {
        int k=(i-1)*m+j;
        if(i!=1) 
          add(k,k-m,val[i][j]+val[i-1][j]),sum+=val[i][j];
        if(i!=n) 
          add(k,k+m,val[i][j]+val[i+1][j]),sum+=val[i][j];
        if(j!=1)
          add(k,k-1,val[i][j]+val[i][j-1]),sum+=val[i][j];
        if(j!=m)
          add(k,k+1,val[i][j]+val[i][j+1]),sum+=val[i][j];
      }
    while(bfs(S,T))
     while (s=dfs(S,T,0x7fffffff))
      mincut+=s;
    ans=sum-mincut;
    printf("%lld\n",ans);
    return 0;
}
### 回答1: bzoj作为一个计算机竞赛的在线评测系统,不仅可以提供大量的题目供程序员练习和学习,还可以帮助程序员提升算法和编程能力。为了更好地利用bzoj进行题目的学习和刷题,制定一个bzoj做题计划是非常有必要的。 首先,我们需要合理安排时间,每天留出一定的时间来做bzoj的题目。可以根据自己的时间安排,每天挑选适量的题目进行解答。可以先从难度较低的题目开始,逐渐提高难度,这样既能巩固基础知识,又能挑战自己的思维能力。 其次,要有一个计划和目标。可以规划一个每周或每月的题目数量目标,以及每个阶段要学习和掌握的算法知识点。可以根据bzoj的题目分类,如动态规划、图论、贪心算法等,结合自己的实际情况,有针对性地选择题目进行学习。 此外,要充分利用bzoj提供的资源。bzoj网站上有很多高质量的题解和优秀的解题代码,可以参考和学习。还有相关的讨论区,可以与其他程序员交和讨论,共同进步。 最后,要坚持并保持思考。做题不是单纯为了刷数量,更重要的是学会思考和总结。遇到难题时,要有耐心,多思考,多尝试不同的解法。即使不能一次性解出来,也要学会思考和分析解题过程,以及可能出现的错误和优化。 总之,bzoj做题计划的关键在于合理安排时间、制定目标、利用资源、坚持思考。通过有计划的刷题,可以提高算法和编程能力,并培养解决问题的思维习惯,在计算机竞赛中取得更好的成绩。 ### 回答2: bzoj做题计划是指在bzoj这个在线测评系统上制定一套学习和刷题的计划,并且将计划记录在excel表格中。该计划主要包括以下几个方面的内容。 首先是学习目标的设定。通过分析自己的水平和知识缺口,可以设定一个合理的目标,比如每天解决一定数量的题目或者提高特定的算法掌握程度。 其次是题目选择的策略。在excel表格中可以记录下自己选择的题目编号、题目类型和难度等信息。可以根据题目的类型和难度来安排每天的刷题计划,确保自己可以逐步提高技巧和解题能力。 然后是学习进度的记录和管理。将每天的完成情况记录在excel表格中,可以清晰地看到自己的学习进度和任务完成情况。可以使用图表等功能来对学习进度进行可视化展示,更好地管理自己的学习计划。 同时,可以在excel表格的备注栏中记录下每道题目的解题思路、关键点和需要复习的知识点等信息。这样可以方便自己回顾和总结,巩固所学的知识。 最后,可以将excel表格与其他相关资料进行整合,比如算法教材、题目解析和学习笔记等。这样可以形成一个完整的学习档案,方便自己进行系统的学习和复习。 总之,bzoj做题计划excel的制定和记录可以帮助我们更加有条理和高效地进行学习和刷题。通过合理安排学习目标和题目选择策略,记录学习进度和思路,并整合其他学习资料,我们可以提高自己的解题能力,并在bzoj上取得更好的成绩。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值