[bzoj2150] 部落战争 二分图

2150: 部落战争

Time Limit: 10 Sec Memory Limit: 259 MB
Submit: 951 Solved: 533
[Submit][Status][Discuss]
Description

lanzerb的部落在A国的上部,他们不满天寒地冻的环境,于是准备向A国的下部征战来获得更大的领土。 A国是一个M*N的矩阵,其中某些地方是城镇,某些地方是高山深涧无人居住。lanzerb把自己的部落分成若干支军队,他们约定: 1. 每支军队可以从任意一个城镇出发,并只能从上往向下征战,不能回头。途中只能经过城镇,不能经过高山深涧。 2. 如果某个城镇被某支军队到过,则其他军队不能再去那个城镇了。 3. 每支军队都可以在任意一个城镇停止征战。 4. 所有军队都很奇怪,他们走的方法有点像国际象棋中的马。不过马每次只能走1*2的路线,而他们只能走R*C的路线。 lanzerb的野心使得他的目标是统一全国,但是兵力的限制使得他们在配备人手时力不从心。假设他们每支军队都能顺利占领这支军队经过的所有城镇,请你帮lanzerb算算至少要多少支军队才能完成统一全国的大业。
这里写图片描述
Input

第一行包含4个整数M、N、R、C,意义见问题描述。接下来M行每行一个长度为N的字符串。如果某个字符是’.’,表示这个地方是城镇;如果这个字符时’x’,表示这个地方是高山深涧。

Output

输出一个整数,表示最少的军队个数。

Sample Input

【样例输入一】

3 3 1 2

.x.

【样例输入二】

5 4 1 1

….

..x.

…x

….

x…

Sample Output

【样例输出一】

4

【样例输出二】

5

【数据范围】

100%的数据中,1<=M,N<=50,1<=R,C<=10。

把一个点拆成两个,一个作为起点,一个作为终点(+cnt)
在每个起点和可以到达的点之间连边,构成一个二分图
最终每个点都要被到达,所以所有的起点都可以派兵去攻占它可以到达的终点
对于每一个终点,如果有边到达它并且从这个边有军队经过,就不需要从这个点开始派兵,这个点是没有花费的
统计出所有没有花费的点,其他的点就是有花费的
没有花费的点数为这个二分图的匹配边数

#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;
const int N = 200 + 5;
const int MAX = 40000 + 5;
int m, n, r, c, cnt;
char s[N][N];
int head[MAX], next[MAX], to[MAX], tail;
bool vis[MAX];
int line[MAX],num[N][N];
void adde( int u, int v ) {
    next[++tail] = head[u];
    head[u] = tail;
    to[tail] = v;
}
int DFS( int u ) {//二分图
    for( int i = head[u]; i; i = next[i] ) {
        int v = to[i];
        if( !vis[v] ){
            vis[v] = true;
            if( !line[v] || DFS(line[v]) ) {
                line[v] = u;
                return 1;
            }
        }
    }
    return 0;
}
int main() {
    scanf("%d%d%d%d", &m, &n, &r, &c);
    for( int i = 1; i <= m; i++ ) {
        scanf("%s", s[i] + 1);
        for( int j = 1; j <= n; j++ )
            if( s[i][j] == '.' )
                num[i][j] = ++cnt;
    }
    for( int i = 1; i <= m; i++ )
        for( int j = 1; j <= n; j++ ) {
            if( !num[i][j] ) continue;
            if(i + r > 0 && j + c > 0 && num[i + r][j + c])   
                adde(num[i][j], num[i + r][j + c] + cnt);
            if(i + c > 0 && j + r > 0 && num[i + c][j + r])
                adde(num[i][j], num[i + c][j + r] + cnt);
            if(i + r > 0 && j - c > 0 && num[i + r][j - c])
                adde(num[i][j], num[i + r][j - c] + cnt);
            if(i + c > 0 && j - r > 0 && num[i + c][j - r])
                adde(num[i][j], num[i + c][j - r] + cnt);
        }
    int ans = 0;
    for( int i = 1; i <= cnt; i++ ) {
        memset(vis,0,sizeof(vis));
        ans += DFS(i);
    }
    printf("%d\n", cnt - ans);
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值