以防万一,题目原文和链接均附在文末。那么先是题目分析:

【一句话题意】

给定大小的棋盘中部分格子存在可以阻止互相攻击的墙,问棋盘中可以放置最多多少个可以横纵攻击炮塔。

【题目分析】

这题本来在搜索专题里出现的..这回又在二分查找匹配专题出现了..所以当然要按照二分匹配的方法解而不是爆搜(虽然爆搜能过)。
问题主要就是如何缩点建图。为了使得blockhouse不能互相攻击,那么使用每行的相邻的点缩点,每列的相邻的点缩点,连边的条件就是两个点存在有相交的部分,最后这两组点求最大匹配就行了。

【算法流程】

应该这题也能算是标准题了吧。缩点,建图,hungary匈牙利算法求解,输出答案。完事儿了。
匈牙利算法以及二分最大匹配的相关内容这里就不说了。。

下面的代码简单说明,本来的aleft和aright名字叫left和right,后来发现和STL的东西冲突了。。
fill的宏定义有更科学的方法,这里就只是单纯的抹0。以及for each是从1到n而不是从0到n。
因为习惯问题,数组是从1算起的。

 #include <cstdio>
#include <cstdlib>
#include <iostream>
#include <cstring>
#include <queue>
#include <algorithm>
#include <vector>
#define each(i,n) (int i=1;i<=(n);++i)
#define fill(arr) memset(arr,0,sizeof(arr))
#define INF 0x3f3f3f3f char map[][];
char aleft[][];
char aright[][];
int l[], r[], lCnt, rCnt;
bool graph[][];
bool vis[]; using namespace std; int dfs(int left) {
for each(right,rCnt) {
//left到right有路且没遍历过
if(graph[left][right] && !vis[right]) {
vis[right] = true;
//若right还没匹配过或跟right匹配的点找到另一个相匹配的点
//(则right就可以跟left匹配)
if(r[right]==- || dfs(r[right])) {
r[right] = left;
l[left] = right;
//printf("(%d,%d)\n",left,right);
return ;
}
}
}
return ;
} int hungary() {
int ans = ;
memset(l,-,sizeof(l));
memset(r,-,sizeof(r));
for each(i,lCnt) { //row point cnt
if(l[i] == -) {
fill(vis);
ans += dfs(i);
}
}
return ans;
} int main() { int n;
char buffer[];
while(gets(buffer)) {
//proc input
if (buffer[] == '') break;
sscanf(buffer,"%d",&n);
for each(line,n) {
gets(buffer);
for each(col,n) {
map[line][col] = buffer[col-];
}
}
//create grapth [ Marking up Points ]
fill(aleft);
fill(aright);
lCnt = ;
int pre = ;
for each(row,n) {
for each(col,n) {
if(map[row][col]=='X') {
if(pre==lCnt) ++lCnt;
continue;
}
aleft[row][col] = lCnt;
pre = lCnt;
}
if(pre==lCnt) ++lCnt;
}
lCnt = pre;
rCnt = ;
pre = ;
for each(col,n) {
for each(row,n) {
if(map[row][col]=='X') {
if(pre==rCnt) ++rCnt;
continue;
}
aright[row][col] = rCnt;
pre = rCnt;
}
if(pre==rCnt) ++rCnt;
}
rCnt = pre;
//create grapth [ Shrinking Points ]
fill(graph);
for each(row,n) {
for each(col,n) {
if(map[row][col]=='.')
graph[aleft[row][col]][aright[row][col]] = true;
}
}
//doWork
printf("%d\n",hungary());
} }

题目链接:Fire Net(HDU 1045)

题目属性:二分图最大匹配 (如果愿意你可以去爆搜..)

相关题目:2444 1083 1281 2819 2389 4185 poj3020 ...

题目原文:
【desc】
Suppose that we have a square city with straight streets. A map of a
city is a square board with n rows and n columns, each representing a
street or a piece of wall.

A blockhouse is a small castle that has four openings through which
to shoot. The four openings are facing North, East, South, and West,
respectively. There will be one machine gun shooting through each
opening.

Here we assume that a bullet is so powerful that it can run across
any distance and destroy a blockhouse on its way. On the other hand, a
wall is so strongly built that can stop the bullets.

The goal is to place as many blockhouses in a city as possible so
that no two can destroy each other. A configuration of blockhouses is
legal provided that no two blockhouses are on the same horizontal row or
vertical column in a map unless there is at least one wall separating
them. In this problem we will consider small square cities (at most 4x4)
that contain walls through which bullets cannot run through.

The following image shows five pictures of the same board. The first
picture is the empty board, the second and third pictures show legal
configurations, and the fourth and fifth pictures show illegal
configurations. For this board, the maximum number of blockhouses in a
legal configuration is 5; the second picture shows one way to do it, but
there are several other ways.

Your task is to write a program that, given a description of a map, calculates the maximum number of blockhouses that can be placed in the city in a legal configuration.
【In】
The input file contains one or more map descriptions, followed by a line
containing the number 0 that signals the end of the file. Each map
description begins with a line containing a positive integer n that is
the size of the city; n will be at most 4. The next n lines each
describe one row of the map, with a '.' indicating an open space and an
uppercase 'X' indicating a wall. There are no spaces in the input file.
【Out】
For each test case, output one line containing the maximum number of
blockhouses that can be placed in the city in a legal configuration.

【SampIn】
4
.X..
....
XX..
....
2
XX
.X
3
.X.
X.X
.X.
3
...
.XX
.XX
4
....
....
....
....
0
【SampOut】
5
1
5
2
4

HDU 1045(Fire Net)题解的更多相关文章

  1. HDOJ(HDU).1045 Fire Net (DFS)

    HDOJ(HDU).1045 Fire Net [从零开始DFS(7)] 点我挑战题目 从零开始DFS HDOJ.1342 Lotto [从零开始DFS(0)] - DFS思想与框架/双重DFS HD ...

  2. hdu 1045 Fire Net(最小覆盖点+构图(缩点))

    http://acm.hdu.edu.cn/showproblem.php?pid=1045 Fire Net Time Limit:1000MS     Memory Limit:32768KB   ...

  3. HDU 1045 Fire Net 状压暴力

    原题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1045 Fire Net Time Limit: 2000/1000 MS (Java/Others)  ...

  4. HDU 1045 Fire Net 【连通块的压缩 二分图匹配】

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=1045 Fire Net Time Limit: 2000/1000 MS (Java/Others)    ...

  5. HDU 1045 Fire Net(dfs,跟8皇后问题很相似)

    传送门:http://acm.hdu.edu.cn/showproblem.php?pid=1045 Fire Net Time Limit: 2000/1000 MS (Java/Others)   ...

  6. HDU 1045——Fire Net——————【最大匹配、构图、邻接矩阵做法】

    Fire Net Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Submit Sta ...

  7. HDU 1045 Fire Net 二分图建图

    HDU 1045 题意: 在一个n*n地图中,有许多可以挡住子弹的墙,问最多可以放几个炮台,使得炮台不会相互损害.炮台会向四面发射子弹. 思路: 把行列分开做,先处理行,把同一行中相互联通的点缩成一个 ...

  8. HDU 1045 - Fire Net - [DFS][二分图最大匹配][匈牙利算法模板][最大流求二分图最大匹配]

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=1045 Time Limit: 2000/1000 MS (Java/Others) Mem ...

  9. hdu 1045 Fire Net(二分图)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=1045 题目大意为给定一个最大为4*4的棋盘,棋盘可以放置堡垒,处在同一行或者同一列的堡垒可以相互攻击, ...

随机推荐

  1. .NET MySQL的参数化查询

    MySqlConnection conn = new MySqlConnection(SqlConnnectString); MySqlCommand cmd = new MySqlCommand(& ...

  2. django配置

    安装python环境后,安装pip工具 通过pip下载安装django pip install django   django在web中的应用主要由两部分构成,工程与App 工程即相当于一下门户框架 ...

  3. 通过递归方法对一个单词所有的组合进行列举(java)

    import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; public ...

  4. 类 的继承性(Inherits)与 重写(Overrides)

    (类) 与 (结构) 类似,让我们可以定义并封装成一组相关项的数据类型.比如封装成结构,那么这个封装包的数据类型就为值类型:如封装成类,那么这个封装包的数据类型就为引用类型. 然而与结构的一个重要区别 ...

  5. (转+原)VC编译错误:uafxcw.lib(afxmem.obj) : error LNK2005: "void * __cdecl operator new(unsigned int)" (??2@YAPAXI@Z) 已经在 LIBCMT.lib(new.obj) 中定义

    参考网址:http://zhanyonhu.blog.163.com/blog/static/16186044201023094754832/ 1>uafxcw.lib(afxmem.obj) ...

  6. mysql数据备份和还原命令

    mysql数据库备份和还原   备份MySQL数据库的命令 mysqldump -hhostname -uusername -ppassword databasename > backupfil ...

  7. windows系统部署discuz并和javaweb账号连通同步

    一.Discuz安装说明 1.安装wamp集成环境 (1)下载wampserver集成环境 网址:(http://wampserver-64bit.en.softonic.com)或百度搜索下载 (2 ...

  8. c#局域网聊天软件的实现

    本软件是基于大学寝室局域网聊天的思路.c#源代码如下: using System; using System.Drawing; using System.Collections; using Syst ...

  9. Python之路第六天,基础(7)-正则表达式(re)

    Python RE模块(正则表达式) 就其本质而言,正则表达式(或 RE)是一种小型的.高度专业化的编程语言,(在Python中)它内嵌在Python中,并通过 re 模块实现.正则表达式模式被编译成 ...

  10. 绑定事件导致发送多个ajax请求的问题

    如果遇到每次触发click, change等事件时ajax请求都会增加一个,这种情况很有可能是每次触发事件都会多绑定一个事件,解决办法就是在每次绑定事件之前要解除绑定.