题目链接:https://ac.nowcoder.com/acm/problem/14698
题目描述
齐齐和司机在玩单机游戏《红色警戒IV》,现在他们的游戏地图被划分成一个n*m的方格地图。齐齐的基地在最上方的4行格内,司机的基地在最下方的4行格内。他们只有一种攻击方式:远程大炮,相关属性如下:
1、 大炮可以打到地图的任意一个位置。
2、 双方每次必须动用本方的一门大炮攻击,齐齐先手,双方交替进行攻击。
3、 一方大炮只能攻击另一方大炮,不能攻击本方或强制攻击未获得视野的地区。
4、 被一方大炮击中的另一方大炮会产生以攻击点为中心的33的波及区域,波及区域内如果有其他大炮则也会产生33的波及区域。
5、 两方的基地相距很远,所以不存在攻打敌方大炮时波及到本方大炮的情况。
齐齐偷偷开了“间谍卫星”,所以他能看到司机的大炮部署,司机则无视野。但如果齐齐做出攻击,司机会立即获取到发动攻击的大炮的视野,并在回合开始时动用大炮(如果存在的话)将其摧毁(摧毁后可能产生的连锁不计入视野)。
现在给出齐齐和司机的大炮部署,问齐齐在选择最优的策略下,在摧毁所有司机的大炮后可以保留最多几门本方大炮。输入描述
第1行输入一个整数m,表示地图的宽度。
第2-5行,每行输入一串长度为m的字符串,代表司机的大炮部署。(大炮为”*“号,空地为“.”号)
第6-9行,每行输入一串长度为m的字符串,代表齐齐的大炮部署。(大炮为”*“号,空地为“.”号)
数据保证:0<m≤100输出描述
输出一行,一个整数。代表摧毁所有司机的大炮后最多保留几门大炮。如果不能摧毁所有司机的大炮,则输出-1。
示例
输入
3
…
.*.
..*
*..
*..
.**
…
*.*输出
4
分析
本题为连通块问题,此处采用深搜解决连通块。 由题意得每门大炮之间有3*3的空间约束,满足该约束条件的所有大炮形成一个块。我们可以从一个块出发向周围8个块进行深度搜索,从而得知一个块的大炮数目。按照题意,齐齐某个块内的大炮向司机某个块中的大炮发起进攻,结果是两个块都将消失。由此我们便可想到,若齐齐的块数目小于司机,则齐齐输,否则,我们便选择使用齐齐的大炮数目小的块去进攻从而留下数目大的块。
因此,在得到两人连通块数据后进行排序,按照上述思路处理即可。AC代码
#include <iostream> #include <vector> #include <queue> #include <string> #include <cstring> #include <algorithm> using namespace std; int m; bool base1[4][100]; //记录司机部署 bool base2[4][100]; //记录齐齐部署 int cnt; bool visited[4][100]; //记录是否搜索过某个位置 int dir[8][2] = {{1, 0}, {-1, 0}, {0, 1}, {0, -1},{1,1},{1,-1},{-1,-1},{-1,1}}; //试探的八个方向 void dfs1(int x, int y) { cnt++; visited[x][y] = true; for (int i = 0; i < 8; i++) { int newx = x + dir[i][0]; int newy = y + dir[i][1]; if (newx < 0 || newx >= 4 || newy < 0 || newy >= m) continue; if (visited[newx][newy]) continue; if(base1[newx][newy]) dfs1(newx,newy); } } void dfs2(int x, int y) { cnt++; visited[x][y] = true; for (int i = 0; i < 8; i++) { int newx = x + dir[i][0]; int newy = y + dir[i][1]; if (newx < 0 || newx >= 4 || newy < 0 || newy >= m) continue; if (visited[newx][newy]) continue; if(base2[newx][newy]) dfs2(newx,newy); } } int main() { cin >> m; cin.get(); for (int i = 0; i < 4; i++) { string temp; getline(cin, temp); for (int j = 0; j < m; j++) { if (temp[j] == '*') base1[i][j] = true; else base1[i][j] = false; } } for (int i = 0; i < 4; i++) { string temp; getline(cin, temp); for (int j = 0; j < m; j++) { if (temp[j] == '*') base2[i][j] = true; else base2[i][j] = false; } } memset(visited,0,sizeof(visited)); vector<int> sum1; //储存各个连通块内大炮的数量 for(int i=0;i<4;i++) { for(int j=0;j<m;j++) { if(base1[i][j] && !visited[i][j]) { cnt=0; dfs1(i,j); sum1.push_back(cnt); } } } memset(visited,0,sizeof(visited)); vector<int> sum2; for(int i=0;i<4;i++) { for(int j=0;j<m;j++) { if(base2[i][j] && !visited[i][j]) { cnt=0; dfs2(i,j); sum2.push_back(cnt); } } } sort(sum1.begin(),sum1.end()); sort(sum2.begin(),sum2.end()); if(sum1.size()>sum2.size()) cout << -1 << endl; else { int sum=0; for(int i=sum1.size()-1;i<sum2.size();i++) sum+=sum2[i]; cout << sum << endl; } return 0; }