热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

HDU5093Battleships(二分图最大匹配)

题意:一个m行n列的图由#、*、o三种符号组成,分别代表冰山、海域、浮冰,问最多可放的炮舰数(要求满足以下条件)

题意:一个m行n列的图由#、*、o三种符号组成,分别代表冰山、海域、浮冰,问最多可放的炮舰数(要求满足以下条件)

1、炮舰只可放在海域处

2、两个炮舰不能放在同一行或同一列(除非中间隔着一个或多个冰山)

分析:

1、如果单纯只考虑不能放在同一行同一列,那就是行号与列号的匹配,原理与UVALive 6811 Irrigation Line(二分图最小点覆盖--匈牙利算法)相同。

2、但现在隔着冰山可以放置炮舰,那假设某一行被冰山分隔成两部分,这一行的前半部分和后半部分可以看做是两行,再应用“行号”与“列号”的匹配的原理

3、既然要行号与列号匹配,那就要分别按照行和列给图标号

举例如下:

(1)若按行,则将海域从1开始标号,如果是连续的海域那标号相同,若有冰山分隔,标号加1.

PS:例如第三行22#3,此处2是形式意义上的第二行,因为有上述放置炮舰的原则,可以理解为是可以放置炮舰的第2行,而由于有冰山分隔,所以这一行的前后互不干扰,因此3可以理解为是可以放置炮舰的第3行

(2)按列同理

4、理论上讲海域可以放置炮舰,也就是说,有标号的地方可以放置炮舰,每一个放置的地方都是一个行号与列号的匹配,由此可得如下匹配

 

5、一条连线代表一个可放置的炮舰,求最大匹配即可

#include
#include

#include

#include

#include

#include

#include

#include

#include

#include
<string>
#include

#include
<set>
#include

#include

#include

#include

#include

#define Min(a, b) a #define Max(a, b) a
typedef long long ll;
typedef unsigned
long long llu;
const int INT_INF &#61; 0x3f3f3f3f;
const int INT_M_INF &#61; 0x7f7f7f7f;
const ll LL_INF &#61; 0x3f3f3f3f3f3f3f3f;
const ll LL_M_INF &#61; 0x7f7f7f7f7f7f7f7f;
const int dr[] &#61; {0, 0, -1, 1};
const int dc[] &#61; {-1, 1, 0, 0};
const double pi &#61; acos(-1.0);
const double eps &#61; 1e-8;
const int MAXN &#61; 50 &#43; 10;
const int MAXT &#61; 2500 &#43; 10;
using namespace std;
char a[MAXN][MAXN];
int x[MAXN][MAXN];
int y[MAXN][MAXN];
int mp[MAXT][MAXT];
bool used[MAXT];
int match[MAXT];
int m, n, cnt1, cnt2;
void deal_row(){cnt1 &#61; 1;for(int i &#61; 0; i i){bool flag &#61; false;for(int j &#61; 0; j j){if(a[i][j] &#61;&#61; &#39;*&#39;){x[i][j] &#61; cnt1;flag &#61; true;}if(a[i][j] &#61;&#61; &#39;#&#39;){&#43;&#43;cnt1;flag &#61; false;}}if(flag) &#43;&#43;cnt1;}
}
void deal_column(){cnt2 &#61; 1;for(int i &#61; 0; i i){bool flag &#61; false;for(int j &#61; 0; j j){if(a[j][i] &#61;&#61; &#39;*&#39;){y[j][i] &#61; cnt2;flag &#61; true;}if(a[j][i] &#61;&#61; &#39;#&#39;){&#43;&#43;cnt2;flag &#61; false;}}if(flag) &#43;&#43;cnt2;}
}
bool Find(int x){for(int i &#61; 1; i i){if(mp[x][i] && !used[i]){used[i] &#61; true;if(!match[i] || Find(match[i])){match[i] &#61; x;return true;}}}return false;
}
void solve(){int cnt &#61; 0;for(int i &#61; 1; i i){memset(used, false, sizeof used);if(Find(i)) &#43;&#43;cnt;}printf("%d\n", cnt);
}
int main(){int T;scanf("%d", &T);while(T--){memset(a, 0, sizeof a);memset(x, 0, sizeof x);memset(y, 0, sizeof y);memset(mp, 0, sizeof mp);memset(match, 0, sizeof match);scanf("%d%d", &m, &n);for(int i &#61; 0; i i)scanf("%s", a[i]);deal_row();deal_column();for(int i &#61; 0; i i){for(int j &#61; 0; j j){if(a[i][j] &#61;&#61; &#39;*&#39;){mp[x[i][j]][y[i][j]] &#61; 1;}}}solve();}return 0;
}

 

转:https://www.cnblogs.com/tyty-Somnuspoppy/p/6036855.html



推荐阅读
author-avatar
金裕麟雯茂俊佑
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有