[백준] 19236번 : 청소년 상어
문제
풀이
할 일을 정리해보자...
1. 상어가 위치한 곳에 있는 물고기를 먹음
1.1 상어의 위치, 방향 정보 갱신
1.2 물고기 삭제 처리
1.3 상어의 새로운 위치를 공간에 기록
2. 1~16번 물고기 이동(아직 살아있는 물고기만)
3. 상어가 이동할 수 있는 모든 공간 탐색(백트래킹 재귀 함수)
3.1 상어가 더 이상 이동할 수 없으면 최댓값 갱신
공간의 현재 상태를 기록할 4x4 2차원 배열과 상어+물고기의 위치, 방향 정보를 기록할 1차원 배열이 필요하다.
상어는 여러 칸을 이동할 수 있지만 방향을 바꿀 수 없고, 물고기는 한 칸만 이동할 수 있지만 방향을 바꿀 수 있다는 차이점을 염두해야 한다.
소스코드
#include <iostream>
#include <vector>
using namespace std;
struct info {
int r, c, d;
};
int max_fish = 0;
pair<int, int> direction[8] = {{-1, 0}, {-1, -1}, {0, -1}, {1, -1}, {1, 0}, {1, 1}, {0, 1}, {-1, 1}};
void moveFish(int idx, vector<info> &pos, vector<vector<int>> &board) {
int row = pos[idx].r, col = pos[idx].c, dir = pos[idx].d; //행, 열, 방향
do {
int nr = row + direction[dir].first;
int nc = col + direction[dir].second;
if ((nr >= 0) && (nr < 4) && (nc >= 0) && (nc < 4) && (board[nr][nc] != 0)) { //범위내에 있고 상어 없으면
swap(board[row][col], board[nr][nc]); //위치 교환
pos[idx] = {nr, nc, dir}; //이번에 이동한 물고기 정보 갱신
if (board[row][col] != -1) //빈공간으로 이동한게 아니라면(물고기와 위치를 바꾼거라면)
pos[board[row][col]] = {row, col, pos[board[row][col]].d}; //바뀐 물고기 정보 갱신
break;
}
dir = (dir + 1) % 8; //방향 바꾸기
} while (dir != pos[idx].d);
}
void backtracking(int r, int c, int fish, vector<info> pos, vector<vector<int>> board) {
fish += board[r][c]; //물고기 먹음
int dir = pos[board[r][c]].d; //먹은 물고기의 방향
pos[0] = {r, c, dir}; //상어 정보 갱신
pos[board[r][c]].d = -1; //먹힌 물고기 갱신
board[r][c] = 0; //상어 위치 표시
for (int i = 1; i <= 16; i++) {
if (pos[i].d != -1) //먹힌 물고기가 아니라면 이동
moveFish(i, pos, board);
}
int nr = r, nc = c;
while (true) { //범위 벗어날 때까지 해당 방향으로 이동
nr += direction[dir].first;
nc += direction[dir].second;
if ((nr < 0) || (nr >= 4) || (nc < 0) || (nc >= 4)) //범위 벗어남
break;
if (board[nr][nc] != -1) { //물고기가 존재
board[r][c] = -1; //원래 상어가 있던 곳은 빈공간으로
backtracking(nr, nc, fish, pos, board);
}
}
max_fish = max(max_fish, fish); //최댓값 갱신
}
int main() {
vector<vector<int>> board(4, vector<int>(4));
vector<info> pos(17);
int a, b;
for (int i = 0; i < 4; i++) {
for (int j = 0; j < 4; j++) {
cin >> a >> b;
board[i][j] = a;
pos[a] = {i, j, b - 1};
}
}
backtracking(0, 0, 0, pos, board);
cout << max_fish;
}
전체 소스코드다.
struct info {
int r, c, d;
};
int max_fish = 0;
pair<int, int> direction[8] = {{-1, 0}, {-1, -1}, {0, -1}, {1, -1}, {1, 0}, {1, 1}, {0, 1}, {-1, 1}};
물고기, 상어의 정보를 저장할 info 구조체와 먹을 물고기 번호의 최댓값을 저장할 max_fish 변수,
그리고 8개의 방향이다.
vector<vector<int>> board(4, vector<int>(4));
vector<info> pos(17);
int a, b;
for (int i = 0; i < 4; i++) {
for (int j = 0; j < 4; j++) {
cin >> a >> b;
board[i][j] = a;
pos[a] = {i, j, b - 1};
}
}
물고기가 16마리 상어가 1마리니까 pos 벡터는 17만큼 할당하고 board 2차원 벡터는 4x4로 만든다.
그리고 입력을 받는데, 방향이 0~7이 되도록 b-1을 해준다.
backtracking(0, 0, 0, pos, board);
함수 호출...시작점은 (0, 0)이고 먹은 물고기의 수는 0이다.
void backtracking(int r, int c, int fish, vector<info> pos, vector<vector<int>> board) {
fish += board[r][c]; //물고기 먹음
int dir = pos[board[r][c]].d; //먹은 물고기의 방향
pos[0] = {r, c, dir}; //상어 정보 갱신
pos[board[r][c]].d = -1; //먹힌 물고기 갱신
board[r][c] = 0; //상어 위치 표시
for (int i = 1; i <= 16; i++) {
if (pos[i].d != -1) //먹힌 물고기가 아니라면 이동
moveFish(i, pos, board);
}
int nr = r, nc = c;
while (true) { //범위 벗어날 때까지 해당 방향으로 이동
nr += direction[dir].first;
nc += direction[dir].second;
if ((nr < 0) || (nr >= 4) || (nc < 0) || (nc >= 4)) //범위 벗어남
break;
if (board[nr][nc] != -1) { //물고기가 존재
board[r][c] = -1; //원래 상어가 있던 곳은 빈공간으로
backtracking(nr, nc, fish, pos, board);
}
}
max_fish = max(max_fish, fish); //최댓값 갱신
}
원래는 백트래킹에서 unvisited 처리를 할 때 pos와 board를 복사해놓고 다시 쓰는 방법으로 구현할까 했는데, 귀찮아서 그냥 파라미터로 넘기기로 했다.
fish += board[r][c]; //물고기 먹음
int dir = pos[board[r][c]].d; //먹은 물고기의 방향
pos[0] = {r, c, dir}; //상어 정보 갱신
pos[board[r][c]].d = -1; //먹힌 물고기 갱신
board[r][c] = 0; //상어 위치 표시
먼저 물고기를 먹었으니 fish 변수를 갱신하고, 해당 물고기의 방향을 저장해둔다.
그리고 그에 맞춰 상어의 행, 열, 방향 정보를 갱신하고, 먹힌 물고기의 방향을 -1로 바꿔준다. 먹혔다는 뜻이다.
마지막으로 원래 물고기가 있던 자리를 0으로 갱신해서 상어가 있다고 표시한다.
for (int i = 1; i <= 16; i++) {
if (pos[i].d != -1) //먹힌 물고기가 아니라면 이동
moveFish(i, pos, board);
}
모든 물고기에 대해 방향 정보가 -1이 아니라면, 그니까 먹힌 물고기가 아니라면 moveFish를 통해 움직여준다.
void moveFish(int idx, vector<info> &pos, vector<vector<int>> &board) {
int row = pos[idx].r, col = pos[idx].c, dir = pos[idx].d; //행, 열, 방향
do {
int nr = row + direction[dir].first;
int nc = col + direction[dir].second;
if ((nr >= 0) && (nr < 4) && (nc >= 0) && (nc < 4) && (board[nr][nc] != 0)) { //범위내에 있고 상어 없으면
swap(board[row][col], board[nr][nc]); //위치 교환
pos[idx] = {nr, nc, dir}; //이번에 이동한 물고기 정보 갱신
if (board[row][col] != -1) //빈공간으로 이동한게 아니라면(물고기와 위치를 바꾼거라면)
pos[board[row][col]] = {row, col, pos[board[row][col]].d}; //바뀐 물고기 정보 갱신
break;
}
dir = (dir + 1) % 8; //방향 바꾸기
} while (dir != pos[idx].d);
}
방향을 옮길 수 있는 횟수는 최대 8번이다.
int nr = row + direction[dir].first;
int nc = col + direction[dir].second;
if ((nr >= 0) && (nr < 4) && (nc >= 0) && (nc < 4) && (board[nr][nc] != 0)) { //범위내에 있고 상어 없으면
swap(board[row][col], board[nr][nc]); //위치 교환
pos[idx] = {nr, nc, dir}; //이번에 이동한 물고기 정보 갱신
if (board[row][col] != -1) //빈공간으로 이동한게 아니라면(물고기와 위치를 바꾼거라면)
pos[board[row][col]] = {row, col, pos[board[row][col]].d}; //바뀐 물고기 정보 갱신
break;
}
dir = (dir + 1) % 8; //방향 바꾸기
dir 방향으로 1칸 움직인 nr, nc에 대해, 이게 범위 내에 있고 그 자리에 상어가 있는게 아니라면
swap으로 위치를 교환한다. 이동하게 될 공간이 빈 공간이라도 상관없다. 어차피 해당 물고기가 빈 공간으로 이동하게 되면 물고기가 있던 곳이 새로운 빈 공간이 될 것이기 때문에 -1과 swap하는 셈이다.
swap을 하면 board의 상태만 갱신되는 것이니, pos[idx]도 갱신하고 만약 물고기와 자리를 바꾼거라면 바뀐 물고기의 위치 정보도 갱신한다.
만약 범위를 벗어나거나 상어가 있어서 이동하지 못했다면 dir값을 갱신하면서 다음 방향을 살펴본다.
int nr = r, nc = c;
while (true) { //범위 벗어날 때까지 해당 방향으로 이동
nr += direction[dir].first;
nc += direction[dir].second;
if ((nr < 0) || (nr >= 4) || (nc < 0) || (nc >= 4)) //범위 벗어남
break;
if (board[nr][nc] != -1) { //물고기가 존재
board[r][c] = -1; //원래 상어가 있던 곳은 빈공간으로
backtracking(nr, nc, fish, pos, board);
}
}
물고기가 전부 이동했으니 상어를 움직일 차례다.
상어의 방향으로 한칸씩 움직여보면서 물고기가 존재하는 칸이라면 원래 상어가 있던 곳을 빈공간으로 바꿔준 뒤, 새로운 좌표에 대해 재귀함수를 호출한다.
max_fish = max(max_fish, fish); //최댓값 갱신
가능한 모든 칸을 탐색한 뒤엔 최댓값을 갱신한다.