zoukankan      html  css  js  c++  java
  • 【CODEVS】1022 覆盖

    算法】二分图匹配(最大流)

    【题解】对i+j进行奇偶染色,就可以保证相邻两格异色。

    然后就是二分图了,对相邻格子连边跑最大流即可。

    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    using namespace std;
    const int maxn=110,maxN=10100,inf=0x3f3f3f3f;
    struct edge{int from,v,flow;}e[300010];
    int n,m,k,tot=1,first[maxN],p[maxn][maxn],cnt,d[maxN],q[1010],S,T,cur[maxN];
    long long ans;
    void insert(int u,int v,int flow)
    {
        tot++;e[tot].v=v;e[tot].flow=flow;e[tot].from=first[u];first[u]=tot;
        tot++;e[tot].v=u;e[tot].flow=0;e[tot].from=first[v];first[v]=tot;
    }
    bool bfs()
    {
        memset(d,-1,sizeof(d));
        int head=0,tail=1;q[0]=S;d[0]=0;
        while(head!=tail)
         {
             int x=q[head++];if(head>=1001)head=0;
             for(int i=first[x];i;i=e[i].from)
              if(e[i].flow&&d[e[i].v]==-1)
               {
                   d[e[i].v]=d[x]+1;
                   q[tail++]=e[i].v;if(tail>=1001)tail=0;
               }
         }
        return d[T]!=-1;
    }
    int dfs(int x,int a)
    {
        if(x==T||a==0)return a;
        int flow=0,f;
        for(int i=first[x];i;i=e[i].from)
         if(d[e[i].v]==d[x]+1&&(f=dfs(e[i].v,min(a,e[i].flow)))>0)
          {
              e[i].flow-=f;
              e[i^1].flow+=f;
              a-=f;
              flow+=f;
              if(a==0)break;
          }
        return flow;
    }
    int main()
    {
        scanf("%d%d%d",&n,&m,&k);
        S=0;
        for(int i=1;i<=k;i++)
         {
             int u,v;
             scanf("%d%d",&u,&v);
             p[u][v]=-1;
         }
        for(int i=1;i<=n;i++)
         for(int j=1;j<=m;j++)
          {
              if(p[i][j]!=-1)
               {
                   p[i][j]=++cnt;
               }
          }
        T=++cnt;
        for(int i=1;i<=n;i++)
         for(int j=1;j<=m;j++)
          {
              if(p[i][j]!=-1)
               {
                   if((i+j)&1)
                    {
                        if(i>1&&p[i-1][j]>0)insert(p[i][j],p[i-1][j],1);
                       if(i<n&&p[i+1][j]>0)insert(p[i][j],p[i+1][j],1);
                       if(j>1&&p[i][j-1]>0)insert(p[i][j],p[i][j-1],1);
                       if(j<n&&p[i][j+1]>0)insert(p[i][j],p[i][j+1],1);
                       insert(S,p[i][j],1);
                    }
                   else insert(p[i][j],T,1);
               }
          }
        ans=0;
        while(bfs())
         {
             for(int i=0;i<=tot;i++)cur[i]=first[i];
             ans+=dfs(S,inf);
         }
        printf("%lld",ans);
        return 0;
    }
    View Code
  • 相关阅读:
    PHP学习(数组)
    PHP学习(语言结构语句)
    PHP学习(运算符)
    PHP学习(数据类型)
    PHP学习(mysqli函数)
    PHP小知识总结(1)
    buffer的相关小知识
    SQL知识总结(1)
    用JS实线放大镜的效果
    单行中文字和图片的相关height和line-height特性
  • 原文地址:https://www.cnblogs.com/onioncyc/p/6498269.html
Copyright © 2011-2022 走看看