불이난 건물에서 상근이가 탈출 하려고한다. 상근이가 얼마만에 탈출 할 수 있는지 알아보자
일단 탈출 할 수없는 경우가 존재하기 때문에 경건한 마음으로 문제를 풀어보자..
불과 상근이의 이동중 먼저 BFS를 사용해야 하는 부분은 불이다. 불이 먼저 번진 것으로 가정해야 문제의 조건과 일치한다.
시간의 개념이 있기 때문에 다음에 번질 위치를 기억하는 Queue를 사용한다. 문제에서는 불과 상근이 두번이 필요하기 때문에 Queue는 총 4개 사용한다.
불이 먼저 번진 경우를 board에 기록하면서 BFS를 진행한다. 상근이를 board에 기록해도 되지만 순서가 불이 먼저기에 상근이는 board에 기록하지 않는다. 대신 현재 위치를 방문 했는지를 기억하는 check 배열을 사용한다.
준비는 끝났다 구현해보자.
#include <iostream>
#include <queue>
using namespace std;
const int MAX = 1000;
const short posX[4] = { 0, 1, 0, -1 };
const short posY[4] = { 1, 0, -1, 0 };
struct Point {
short x;
short y;
Point() : x(0), y(0) {};
Point(short x, short y) : x(x), y(y) {};
};
int main()
{
ios_base::sync_with_stdio(false);
cin.tie(0);
int tcc;
cin >> tcc;
for (int i = 0; i < tcc; i++)
{
queue<Point> fire;
queue<Point> nextFire;
queue<Point> sangGeun;
queue<Point> nextSangGeun;
int result = 0;
bool success = false;
char board[MAX+1][MAX + 1] = {};
bool check[MAX+1][MAX] = {};
int row, col;
cin >> row >> col;
for (int i = 0; i < col; i++)
{
cin >> board[i];
for (int j = 0; j < row; j++)
if (board[i][j] == '*')
fire.push(Point(i, j));
else if (board[i][j] == '@')
{
sangGeun.push(Point(i, j));
check[i][j] = true;
}
}
while (!sangGeun.empty())
{
while (!fire.empty())
{
Point cur = fire.front();
fire.pop();
for (int i = 0; i < 4; i++)
{
short nextX = cur.x + posX[i];
short nextY = cur.y + posY[i];
if (0 <= nextX && nextX < col && 0 <= nextY && nextY < row && board[nextX][nextY] != '#' && board[nextX][nextY] != '*')
{
board[nextX][nextY] = '*';
nextFire.push(Point(nextX, nextY));
}
}
}
while (!sangGeun.empty())
{
Point cur = sangGeun.front();
sangGeun.pop();
if (cur.x == -1 || cur.y == -1 || cur.x == col || cur.y == row)
{
success = true;
break;
}
// cout << cur.x << ' ' << cur.y << '\n'; // For Debug
for (int i = 0; i < 4; i++)
{
short nextX = cur.x + posX[i];
short nextY = cur.y + posY[i];
if (nextX == -1 || nextY == -1 || nextX == col || nextY == row)
nextSangGeun.push(Point(nextX, nextY));
else if (nextX < col && nextY < row && !check[nextX][nextY] &&board[nextX][nextY] != '#' && board[nextX][nextY] != '*')
{
check[nextX][nextY] = true;
nextSangGeun.push(Point(nextX, nextY));
}
}
}
if (!success)
{
while (!nextFire.empty())
{
fire.push(nextFire.front());
nextFire.pop();
}
while (!nextSangGeun.empty())
{
sangGeun.push(nextSangGeun.front());
nextSangGeun.pop();
}
result++;
}
else
break;
}
if (success)
cout << result << '\n';
else
cout << "IMPOSSIBLE\n";
}
return 0;
}
2019-01-29 20:54:21에 Tistory에서 작성되었습니다.