模拟战役


题目链接: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;
}

文章作者: Kong Aobo
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 Kong Aobo !
  目录