백조의 호수
https://www.acmicpc.net/problem/3197
| 시간 제한 | 메모리 제한 | 제출 | 정답 | 맞힌 사람 | 정답 비율 |
|---|---|---|---|---|---|
| 1 초 | 256 MB | 27610 | 5615 | 3494 | 19.141% |
문제
두 마리의 백조가 호수에서 살고 있었다.
그렇지만 두 마리는 호수를 덮고 있는 빙판으로 만나지 못한다.
호수는 행이 R개, 열이 C개인 직사각형 모양이다.
어떤 칸은 얼음으로 덮여있다.
호수는 차례로 녹는데, 매일 물 공간과 접촉한 모든 빙판 공간은 녹는다.
두 개의 공간이 접촉하려면 가로나 세로로 닿아 있는 것만 (대각선은 고려하지 않는다) 생각한다.
아래에는 세 가지 예가 있다.
...XXXXXX..XX.XXX ....XXXX.......XX .....XX..........
....XXXXXXXXX.XXX .....XXXX..X..... ......X..........
...XXXXXXXXXXXX.. ....XXX..XXXX.... .....X.....X.....
..XXXXX..XXXXXX.. ...XXX....XXXX... ....X......XX....
.XXXXXX..XXXXXX.. ..XXXX....XXXX... ...XX......XX....
XXXXXXX...XXXX... ..XXXX.....XX.... ....X............
..XXXXX...XXX.... ....XX.....X..... .................
....XXXXX.XXX.... .....XX....X..... .................
처음 첫째 날 둘째 날
백조는 오직 물 공간에서 세로나 가로로만(대각선은 제외한다) 움직일 수 있다.
며칠이 지나야 백조들이 만날 수 있는 지 계산하는 프로그램을 작성하시오.
입력
입력의 첫째 줄에는 R과 C가 주어진다. 단, 1 ≤ R, C ≤ 1500.
다음 R개의 줄에는 각각 길이 C의 문자열이 하나씩 주어진다.
‘.’은 물 공간, ‘X’는 빙판 공간, ‘L’은 백조가 있는 공간으로 나타낸다.
출력
첫째 줄에 문제에서 주어진 걸리는 날을 출력한다.
예제 입력 1
10 2 .L .. XX XX XX XX XX XX .. .L
예제 출력 1
3
예제 입력 2
4 11 ..XXX...X.. .X.XXX...L. ....XXX..X. X.L..XXX...
예제 출력 2
2
예제 입력 3
8 17 ...XXXXXX..XX.XXX ....XXXXXXXXX.XXX ...XXXXXXXXXXXX.. ..XXXXX.LXXXXXX.. .XXXXXX..XXXXXX.. XXXXXXX...XXXX... ..XXXXX...XXX.... ....XXXXX.XXXL...
예제 출력 3
2
출처
Olympiad > Croatian Highschool Competitions in Informatics > 2005 > National Competition #2 – Seniors 2번
- 문제를 번역한 사람: baemin0103
- 데이터를 추가한 사람: dlbae9613
알고리즘 분류
통과된 코드
해당 얼음이 언제 녹는지 확인하고 저장합니다.
BFS 탐색을 할 때 목표에 도달하지 못한 생태에서 Queue가 empty라면
날짜를 하루씩 증가시키고 해당 일자에 녹는 얼음을 빈칸으로 변경하고 탐색을 이어나간다.
#include <iostream>
#include <queue>
#include <vector>
using namespace std;
int _R, _C, _DxDy[4][2] = { {1, 0}, {-1, 0}, {0, 1}, {0, -1}};
vector<pair<int, int>> _SwanPosV;
vector<pair<int, int>> _IcePosV[750];
queue<pair<int, pair<int, int>>> _IceQueue;
queue<pair<int, int>> _SwanQueue;
char _Map[1501][1501];
bool _isVisted[1501][1501];
string str;
int main()
{
ios_base::sync_with_stdio(false);
cin.tie(NULL);
cout.tie(NULL);
cin >> _R >> _C;
for (int i = 0; i < _R; i++) {
cin >> str;
for (int j = 0; j < _C; j++) {
_Map[i][j] = str[j];
if (str[j] == 'L') {
_SwanPosV.push_back(make_pair(i, j));
_Map[i][j] = '.';
}
else if (str[j] == 'X')
_IcePosV[0].push_back(make_pair(i, j));
}
}
int _dx, _dy;
pair<int, int> _Pos;
for (auto& it : _IcePosV[0]) {
for (int j = 0; j < 4; j++) {
_dx = it.first + _DxDy[j][0];
_dy = it.second + _DxDy[j][1];
if (_dx >= _R || _dy >= _C || _dx < 0 || _dy < 0)
continue;
if (_Map[_dx][_dy] == '.') {
_IcePosV[1].push_back(it);
break;
}
}
}
for (auto& it : _IcePosV[1]) {
_isVisted[it.first][it.second] = true;
_IceQueue.push(make_pair(1, it));
}
while (!_IceQueue.empty()) {
_Pos = _IceQueue.front().second;
int _cnt = _IceQueue.front().first;
_IceQueue.pop();
for (int j = 0; j < 4; j++) {
_dx = _Pos.first + _DxDy[j][0];
_dy = _Pos.second + _DxDy[j][1];
if (_dx >= _R || _dy >= _C || _dx < 0 || _dy < 0)
continue;
if (_isVisted[_dx][_dy] || _Map[_dx][_dy] != 'X')
continue;
_isVisted[_dx][_dy] = true;
_IceQueue.push(make_pair(_cnt + 1, make_pair(_dx, _dy)));
_IcePosV[_cnt + 1].push_back(make_pair(_dx, _dy));
}
}
for (int i = 0; i < _R; i++)
for (int j = 0; j < _C; j++)
_isVisted[i][j] = false;
_SwanQueue.push(_SwanPosV[0]);
_isVisted[_SwanPosV[0].first][_SwanPosV[0].second] = true;
_IcePosV[0].clear();
int _Day = 0;
while (true) {
if (_SwanQueue.empty()) {
_Day++;
for (auto& it : _IcePosV[_Day])
_Map[it.first][it.second] = '.';
for (auto& it : _IcePosV[0])
_SwanQueue.push(it);
_IcePosV[0].clear();
}
_Pos = _SwanQueue.front();
_SwanQueue.pop();
if (_Pos == _SwanPosV[1])
break;
for (int j = 0; j < 4; j++) {
_dx = _Pos.first + _DxDy[j][0];
_dy = _Pos.second + _DxDy[j][1];
if (_dx >= _R || _dy >= _C || _dx < 0 || _dy < 0)
continue;
if (_isVisted[_dx][_dy])
continue;
if (_Map[_dx][_dy] == 'X') {
_isVisted[_dx][_dy] = true;
_IcePosV[0].push_back(make_pair(_dx, _dy));
continue;
}
_SwanQueue.push(make_pair(_dx, _dy));
_isVisted[_dx][_dy] = true;
}
}
cout << _Day;
return 0;
}

위와 같은 로직으로 진행하면 생각보다 메모리와 시간을 많이 소모한다.
한번의 BFS 탐색으로 문제를 해결하는 방법도 있지만 생각했던 방법으로 구현해보고 싶어서 진행

![백준 10474번 (분수좋아해?, C++) [BAEKJOON]](https://lycos7560.com/wp-content/uploads/2022/10/boj-og-1-2048x1070-1-1024x535.png)
![백준 2042번 (구간 합 구하기, C++) [BAEKJOON]](https://lycos7560.com/wp-content/uploads/boj-og.png)