题意: 给你一副图, 有草地(*),空地(o)和墙(#),空地上可以放机器人, 机器人向上下左右4个方向开枪(枪不能穿墙),问你在所有机器人都不相互攻击的情况下能放的最多的机器人数。

思路:这是一类经典题的衍化,如果没有墙,我们会将行和列看成两列点阵,然后就可以用二分匹配解。

现在有墙怎么办呢, 把某一行或列(有墙的拆分成多个区域,可以看成多个行或列), 拆好以后更没有墙的做法一样了。

#include <cstdio>
#include <cstring>
#include <vector>
#include <algorithm>
using namespace std;
const int maxn = 1505;
vector <int> edge[maxn]; //记录以左排点为起点的单向边
int pre[maxn]; //右点阵的大小
bool vis[maxn]; //右点阵的大小
int n, m;
bool dfs(int u) {
int i, v;
for(i = 0; i < (int)edge[u].size(); i++) {
v = edge[u][i];
if(vis[v])
continue;
vis[v] = 1;
if(pre[v] == -1 || dfs(pre[v])) {
pre[v] = u;
return 1;
}
}
return 0;
}
char mp[51][51];
int num[51][51]; int nx, ny, x[maxn][maxn], y[maxn][maxn];
int main() {
int i, j, cas, ca = 1;
scanf("%d", &cas);
while(cas--) {
scanf("%d%d", &n, &m);
for(i = 0; i < n; i++)
scanf("%s", mp[i]);
memset(x, -1, sizeof(x));
nx = 0;
for(i = 0; i < n; i++) {
for(j = 0; j < m; j++)
if(mp[i][j] == 'o') x[i][j]= nx;
else if(mp[i][j] == '#') nx++;
nx++;
}
memset(y, -1, sizeof(y));
ny = 0;
for(j = 0; j < m; j++) {
for(i = 0; i < n; i++)
if(mp[i][j] == 'o') y[i][j] = ny;
else if(mp[i][j] == '#') ny++;
ny++;
}
for(i = 0; i < nx; i++) edge[i].clear(); for(i = 0; i < n; i++)
for(j = 0; j < m; j++)
if(mp[i][j] == 'o')
edge[x[i][j]].push_back(y[i][j]);
memset(pre, -1, sizeof(int)*ny);
//建边
int cnt = 0;
for(i = 0; i < nx; i++) {
memset(vis, 0, sizeof(int)*ny);
if(dfs(i)) cnt++;
}
printf("Case :%d\n%d\n", ca++, cnt);
}
return 0;
}

最新文章

  1. jquery mobile
  2. 关于移动端swiper的2种样式重置
  3. 2015安徽省赛 G.你来擒孟获
  4. loadRunner录制脚本常见问题及解决方法
  5. windows下安装boost库
  6. SQL 导出表结构到Excel
  7. Implement strStr() [LeetCode]
  8. codeforces 711E E. ZS and The Birthday Paradox(数学+概率)
  9. json 读写 swift
  10. sublime 3 注册码
  11. 如何让oracle的select强制走索
  12. android 应用程序框架
  13. CentOS7 下安装telnet服务
  14. thinkinginjava学习笔记02_对象
  15. Dynamics CRM 检测访问CRM延迟及带宽的工具
  16. 1013. Battle Over Cities (25)
  17. UVA11419 SAM I AM
  18. Silverlight界面设计
  19. lfs(systemd版本)学习笔记-第3页
  20. 关于php-fpm方式和apache配合使用的几点记录

热门文章

  1. C# 未能加载文件或程序集“MySQLDriverCS...&quot; 错误解决
  2. EasyUI - Panel 面板控件
  3. 基于visual Studio2013解决面试题之0503取最大数字字符串
  4. 语音信号处理之(三)矢量量化(Vector Quantization)
  5. 第四天学习内容 if switch for 的练习
  6. Tomcat详细用法学习(五)
  7. CodeForces 370C. Mittens
  8. 如何使用Reaver破解Wi-Fi网络的WPA密码
  9. ANT编译Android Eclipse工程
  10. Linux教程:如何在Linux下进行C++开发?