https://www.acmicpc.net/problem/15558
문제
상근이는 오른쪽 그림과 같은 지도에서 진행하는 게임을 만들었다.
지도는 총 2개의 줄로 나누어져 있으며, 각 줄은 N개의 칸으로 나누어져 있다. 칸은 위험한 칸과 안전한 칸으로 나누어져 있고, 안전한 칸은 유저가 이동할 수 있는 칸, 위험한 칸은 이동할 수 없는 칸이다.
가장 처음에 유저는 왼쪽 줄의 1번 칸 위에 서 있으며, 매 초마다 아래 세 가지 행동중 하나를 해야 한다.
- 한 칸 앞으로 이동한다. 예를 들어, 현재 있는 칸이 i번 칸이면, i+1번 칸으로 이동한다.
- 한 칸 뒤로 이동한다. 예를 들어, 현재 있는 칸이 i번 칸이면, i-1번 칸으로 이동한다.
- 반대편 줄로 점프한다. 이때, 원래 있던 칸보다 k칸 앞의 칸으로 이동해야 한다. 예를 들어, 현재 있는 칸이 왼쪽 줄의 i번 칸이면, 오른쪽 줄의 i+k번 칸으로 이동해야 한다.
N번 칸보다 더 큰 칸으로 이동하는 경우에는 게임을 클리어한 것이다.
게임을 재밌게 하기 위해서, 상근이는 1초에 한 칸씩 각 줄의 첫 칸이 사라지는 기능을 만들었다. 즉, 게임을 시작한지 1초가 지나면 1번 칸이 사라지고, 2초가 지나면 2번 칸이 사라진다. 편의상 유저가 먼저 움직이고, 칸이 사라진다고 가정한다. 즉, 이번에 없어질 칸이 3번 칸인데, 상근이가 3번 칸에 있다면, 3번 칸에서 다른 칸으로 이동하고 나서 3번 칸이 없어지는 것이다.
각 칸의 정보가 주어졌을 때, 게임을 클리어 할 수 있는지, 없는지 구하는 프로그램을 작성하시오.
입력
첫째 줄에 N과 k가 주어진다. (1 ≤ N, k ≤ 100,000)
둘째 줄에는 왼쪽 줄의 정보가 주어진다. i번째 문자가 0인 경우에는 위험한 칸이고, 1인 경우에는 안전한 칸이다. 셋째 줄에는 오른쪽 줄의 정보가 주어지고, 각 문자의 의미는 왼쪽 줄의 의미와 동일하다.
왼쪽 줄의 1번 칸은 항상 안전한 칸이다.
출력
게임을 클리어할 수 있으면 1을, 없으면 0을 출력한다.
문제 풀이
왼쪽 줄과 오른쪽 줄을 구분하여 그래프를 탐색하는 문제.
현재 왼쪽/오른쪽 줄 i번째 칸에 있을 때 오른쪽/왼쪽 줄 i+k칸이 1이라면 옆 줄로 이동할 수 있고, 시간 t가 i일 때 i번째 칸이 사라지는 것에 주의하여 bfs 탐색을 진행하면 된다.
앞 칸( i+1 칸)으로 이동할 때에는 탐색을 진행하지 않은 칸이고 안전한 칸이면 이동 가능하다.
뒷 칸 ( i-1 칸)으로 이동할 때에는 탐색을 진행하지 않은 칸이고 안전한 칸이어야 한다. 또한 i번째 칸의 시간으로부터 1초 지났을 때의 시간이 i-1을 넘지 않아야 이동할 수 있다.
반대편 줄로 점프할 때에는 탐색을 진행하지 않은 칸이고 i+k칸이 안전한 칸이어야 한다.
현재 칸이 N- k칸 이상일 때 반대편 줄로 점프하여 무조건 게임을 클리어 할 수 있으므로 탈출할 수 있다.
아래는 코드.
#include <iostream>
#include <queue>
using namespace std;
int* leftArr;
int* rightArr;
bool* isVisitedLeft;
bool* isVisitedRight;
int N, K;
struct info
{
string pos;
int index;
int time;
info(string a, int b, int c)
{
pos = a;
index = b;
time = c;
}
};
queue<info> bfsQueue;
void bfs()
{
bool canClear = false;
bfsQueue.push(info("left", 0, 0));
isVisitedLeft[0] = true;
while (!bfsQueue.empty())
{
info current = bfsQueue.front();
if (current.index >= N - K)
{
canClear = true;
break;
}
if (current.pos == "left")
{
if (isVisitedLeft[current.index + 1] == false && leftArr[current.index+1] == 1)
{
isVisitedLeft[current.index + 1] = true;
bfsQueue.push(info("left", current.index + 1, current.time + 1));
}
if (isVisitedLeft[current.index - 1] == false && current.time+1 <= current.index-1 &&leftArr[current.index - 1] == 1)
{
isVisitedLeft[current.index - 1] = true;
bfsQueue.push(info("left", current.index - 1, current.time + 1));
}
if (isVisitedRight[current.index + K] == false && rightArr[current.index + K] == 1)
{
isVisitedRight[current.index + K] = true;
bfsQueue.push(info("right", current.index + K, current.time + 1));
}
}
else
{
if (isVisitedRight[current.index + 1] == false && rightArr[current.index + 1] == 1)
{
isVisitedRight[current.index + 1] = true;
bfsQueue.push(info("right", current.index + 1, current.time + 1));
}
if (isVisitedRight[current.index - 1] == false && current.time + 1 <= current.index - 1 && rightArr[current.index - 1] == 1)
{
isVisitedRight[current.index - 1] = true;
bfsQueue.push(info("right", current.index - 1, current.time + 1));
}
if (isVisitedLeft[current.index + K] == false && leftArr[current.index + K] == 1)
{
isVisitedLeft[current.index + K] = true;
bfsQueue.push(info("left", current.index + K, current.time + 1));
}
}
bfsQueue.pop();
}
if (canClear == true)
{
cout << 1 << "\n";
}
else
{
cout << 0 << "\n";
}
}
int main()
{
cin.tie(NULL);
ios::sync_with_stdio(false);
string str;
cin >> N >> K;
leftArr = new int[N];
rightArr = new int[N];
isVisitedLeft = new bool[N];
isVisitedRight = new bool[N];
cin >> str;
for (int i = 0; i < N; i++)
{
leftArr[i] = str.at(i) - 48;
isVisitedLeft[i] = false;
}
cin >> str;
for (int i = 0; i < N; i++)
{
rightArr[i] = str.at(i) - 48;
isVisitedRight[i] = false;
}
bfs();
return 0;
}
'알고리즘 > 백준' 카테고리의 다른 글
[백준 2283] 구간 자르기 (C++) (0) | 2024.03.17 |
---|---|
[백준 20366] 같이 눈사람 만들래? (C++) (0) | 2024.03.16 |
[백준 20365] 블로그2 (C++) (0) | 2024.03.14 |
[백준 1477] 휴게소 세우기 (C++) (0) | 2024.03.13 |
[백준 2910] 빈도 정렬 (C++) (0) | 2024.03.12 |