码迷,mamicode.com
首页 > 其他好文 > 详细

bzoj1066

时间:2016-11-22 23:20:26      阅读:208      评论:0      收藏:0      [点我收藏+]

标签:nic   cin   dinic   turn   code   color   sizeof   while   span   

首先,我们可以想到从源点向每个有蜥蜴的地方连边,然后拆点,因为我们不能把一个点连向多条边,这样修改边的时候不可以,所以拆个点,就可以了

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
const int inf=0x3f3f3f3f;
struct edge
{
    int to,nxt,f;
}e[3011];
string s;
int r,c,d,ans,cnt=1,tot;
int board[1010][1010];
int g[3011],dist[3011];
void link(int u,int v,int f)
{
    e[++cnt].nxt=g[u];
    g[u]=cnt;
    e[cnt].f=f;
    e[cnt].to=v;
}
inline int Min(int x,int y)
{
    return x<y?x:y;
}
inline int abs(int x)
{
    return x<0?-x:x;
}
inline int Max(int x,int y)
{
    return x>y?x:y;
}
inline void ins(int u,int v,int f)
{
    link(u,v,f); link(v,u,0);
}
bool bfs()
{
    queue<int>q;
    memset(dist,inf,sizeof(dist));
    dist[0]=0;
    q.push(0);
    while(!q.empty())
    {
        int u=q.front(); q.pop();
        for(int i=g[u];i;i=e[i].nxt)
        {
            int v=e[i].to;
            if(e[i].f&&dist[v]==inf)
            {
                dist[v]=dist[u]+1;
                q.push(v);
            }
        }
    }
    return dist[1010]!=inf;
}
int dfs(int u,int delta)
{
    if(u==1010) return delta;
    int ret=0;
    for(int i=g[u];i&&delta;i=e[i].nxt)
    {
        int v=e[i].to;
        if(e[i].f&&dist[v]==dist[u]+1)
        {
            int dd=dfs(v,Min(e[i].f,delta));
            delta-=dd;
            e[i].f-=dd;
            e[i^1].f+=dd;
            ret+=dd;
        }
    }
    return ret;
}
inline void dinic()
{
    while(bfs())
    {
        ans+=dfs(0,inf);
    }
    printf("%d",tot-ans);
}
int main()
{
    cin>>r>>c>>d;
    for(int i=1;i<=r;i++)
    {
        cin>>s;
        for(int j=0;j<s.length();j++)
        {
            board[i][j+1]=s[j]-0;
        }
    }
    for(int i=1;i<=r;i++)
        for(int j=1;j<=c;j++) if(board[i][j])
        {
            ins((i-1)*c+j,(i-1)*c+j+r*c,board[i][j]);
            if(j-d<=0||j+d>c||i-d<=0||i+d>r)
            {
                ins((i-1)*c+j+r*c,1010,board[i][j]);
            }
            for(int a=Max(1,i-d);a<=Min(i+d,r);a++)
                for(int b=Max(1,j-d);b<=Min(j+d,c);b++)
                    if(abs(a-i)+abs(b-j)<=d&&board[a][b])
                        ins((i-1)*c+j+r*c,(a-1)*c+b,inf);
        }
    for(int i=1;i<=r;i++)
    {
        cin>>s;
        for(int j=0;j<s.length();j++)
        {
            if(s[j]==L) {tot++; ins(0,(i-1)*c+j+1,1);}        
        }
    }
    dinic();
    return 0;
}

 

bzoj1066

标签:nic   cin   dinic   turn   code   color   sizeof   while   span   

原文地址:http://www.cnblogs.com/19992147orz/p/6091368.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!