메두사와 전사들 [2024-하반기 오후 1번 문제]
문제 링크

문제 설명
N*N
크기의 배열 → 움직일 수 있는 곳 0
, 아닌 곳 1
메두사의 좌표 S_r
, S_c
, 공원(도착지)의 좌표 E_r
, E_c
M
명의 전사(방해물) (r_i,c_i)
에 위치, 각 턴마다 메두사를 향해 이동(도로 아닌 곳도 가능)
- 메두사의 이동최단 경로를 따라 메두사가 이동. 이동한 위치에 전사가 있다면 제거 (상 하 좌 우의 우선순위로 이동)
- 메두사의 시선상 하 좌 우의 방향을 선택해서 바라봄. 시야각에 들어와있지만 다른 전사(방해물)에 가려져 있는 경우 보이지 않음. 시선 안에 위치한 전사(방해물)은 해당 턴에 움직일 수 없음. (같은 위치에 여러 명의 전사(방해물)이 있을 수 있음.) 전사(방해물)을 가장 많이 볼 수 있는 방향으로 바라보며, 같은 수가 여러개라면 상 하 좌 우로 우선순위
- 전사 이동메두사와 가까워지는 방향으로 두번 이동. 격자 밖으로 나갈 수 없고, 메두사의 시야로는 이동할 수 없음. 첫 번째 이동에는 상 하 좌 우의 우선순위이고, 두 번째 이동에는 좌 우 상 하의 우선순위. 메두사의 위치에 도달한 전사는 사망
- 출력물각 턴마다
모든 전사가 이동한 거리의 합, 메두사로 인해 돌로 변한 전사의 수, 메두사가 공격한 전사의 수
를 출력
해결 방법
전사들 이동 및 공격
for (int i = 0;i < m;i++) {
if (warrior[i].died || warrior[i].stone) continue;
warrior_move_cnt += warrior_move(i, path[p].x, path[p].y,true);
warrior_move_cnt += warrior_move(i, path[p].x, path[p].y,false);
}
2번 이동하는데, 상하좌우의 우선순위가 다르기 때문에, bool 값으로 다르게 설정해준다.
int warrior_move(int cur, int mx,int my, bool is_first) {
int dx[4], dy[4];
if (is_first) {
dx[0] = -1, dx[1] = 1, dx[2] = 0, dx[3] = 0;
dy[0] = 0, dy[1] = 0, dy[2] = -1, dy[3] = 1;
}
else {
dx[2] = -1, dx[3] = 1, dx[0] = 0, dx[1] = 0;
dy[2] = 0, dy[3] = 0, dy[0] = -1, dy[1] = 1;
}
int cx, cy, nx, ny;
cx = warrior[cur].x, cy = warrior[cur].y;
int cur_distance = abs(mx - cx) + abs(my - cy);
int next_distance = 987654321;
int nd = -1;
for (int d = 0;d < 4;d++) {
int tmp_distance;
nx = cx + dx[d], ny = cy + dy[d];
if (nx < 0 || ny < 0 || nx >= n || ny >= n) continue;
if (sight_arr[nx][ny] == 1) continue;
tmp_distance = abs(mx - nx) + abs(my - ny);
if (tmp_distance >= cur_distance || tmp_distance >= next_distance) continue;
next_distance = tmp_distance;
nd = d;
}
if (nd == -1) return 0;
warrior[cur].x += dx[nd], warrior[cur].y += dy[nd];
return 1;
}
메두사가 병사를 돌로 만드는 함수에서 미리 선언한 sight_arr를 활용하여, 해당 병사가 메두사의 시야에 있는지, 격자를 벗어나는지 확인하고 최단거리를 계산해서 돌려준다.
int died_count = 0;
for (int i = 0;i < m;i++) {
if (warrior[i].x == path[p].x && warrior[i].y == path[p].y && !warrior[i].died && !warrior[i].stone) {
warrior[i].died = true;
died_count++;
}
warrior[i].stone = false;
}
이동한 병사들의 위치를 확인하여 메두사와 같은 위치라면 죽음으로 변경하고, 출력을 위하여 died_count
변수에 죽은 인원을 추가한다.
가장 많은 전사를 바라볼 수 있는 시야 선정 및 돌로 만들기
int d = -1; int stone_chk = 0;
for (int i = 0;i < 4;i++) {
int cur_val = medu_sight(i, path[p].x, path[p].y,false);
if (cur_val > stone_chk) {
d = i, stone_chk = cur_val;
}
}
medu_sight
함수를 이용하여, 각 위치에서 메두사가 봐야 할 방향을 d
에 저장하고, 출력을 위한 돌로 된 병사들의 수를 stone_chk
에 저장한다.
int dx[4] = { -1,1,0,0 };
int dy[4] = { 0,0,-1,1 };
memset(marr, 0, sizeof(marr));
// 1. 병사들 위치 체크
for (int i = 0;i < m;i++) {
if (warrior[i].died) continue;
marr[warrior[i].x][warrior[i].y]++;
}
// 메두사 시야 설정을 위한 변수
int cnt = 1;
int s, e;
int ret = 0;
if (d == 0) { // 상단
for (int i = x - 1;i >= 0;i--) {
s = y - cnt < 0 ? 0 : y - cnt;
e = y + cnt >= n ? n - 1 : y + cnt;
for (int j = s;j <= e;j++) {
if (real_flag && sight_arr[i][j] != -1) sight_arr[i][j] = 1;
if (marr[i][j] > 0) { // 병사가 있다면
set_war_back(i, j, x, y, d,real_flag);
ret += marr[i][j];
marr[i][j] = 999;
}
}
cnt++;
}
}
쉬운 조건 탐색을 위해, 우선 marr
배열에 병사의 위치를 표시해놓는다. 메두사는 이등변삼각형 모양의 시야를 갖기 때문에, 이중 배열 및 추가 변수를 활용하여 메두사가 볼 수 있는 병사들을 체크한다. 만약, 메두사가 병사를 발견하면 해당 위치의 marr
의 값을 999
로 변경하고, 사각 지대(해당 병사의 뒤)를 set_war_back
함수를 통해 음수로 변경한다.
int wloc;
int cnt = 1;
int s, e;
if (d == 0) {
wloc = my - y;
for (int i = x - 1;i >= 0;i--) {
if (wloc > 0) {
s = y - cnt < 0 ? 0 : y - cnt;
e = y;
}
else if (wloc < 0) {
s = y;
e = y + cnt >= n ? n - 1 : y + cnt;
}
else {
s = y, e = y;
}
for (int j = s;j <= e;j++) {
marr[i][j] = -1;
if (real_flag) sight_arr[i][j] = -1;
}
cnt++;
}
}
wloc
을 방향 당 조건으로 설정한다. (wloc
이 양수이면 좌측, 0이면 뒤에만, 음수면 우측 … 이런식으로) 해당 조건을 활용하여, 병사 뒤의 값들을 -1
로 변경한다.
if (real_flag) {
for (int i = 0;i < m;i++) {
if (marr[warrior[i].x][warrior[i].y] == 999) {
warrior[i].stone = true;
}
}
}
return ret;
d
를 얻고 나면, real_flag
인자를 true
로 넣어 999
(발각된 병사들)의 위치에 해당하는 병사들의 stone
여부를 true
로 변경한다.
메두사의 이동경로 저장get_path
함수에 해당하는 과정임.
queue<pair<int, int>> que;
que.push({ start[0],start[1] });
visited[start[0]][start[1]] = 1;
int dx[4] = { -1,1,0,0 };
int dy[4] = { 0,0,-1,1 };
bool found = false;
int cx, cy;
while (!que.empty()) {
tie(cx, cy) = que.front();
que.pop();
if (cx == dest[0] && cy == dest[1]) {
found = true;
break;
}
for (int i = 0;i < 4;i++) {
int nx, ny;
nx = cx + dx[i], ny = cy + dy[i];
if (nx < 0 || ny < 0 || nx >= n || ny >= n) continue;
if (visited[nx][ny] || arr[nx][ny]) continue;
visited[nx][ny] = 1;
parent[nx][ny][0] = cx, parent[nx][ny][1] = cy;
que.push({ nx,ny });
}
}
if (!found) return false;
BFS
를 활용해서 start
에서 dest
까지의 경로를 parent
배열에 저장
cx = dest[0], cy = dest[1];
while (!(cx == start[0] && cy == start[1])) {
_loc tmp;
tmp.x = cx, tmp.y = cy;
path.push_back(tmp);
cx = parent[tmp.x][tmp.y][0], cy = parent[tmp.x][tmp.y][1];
}
reverse(path.begin(), path.end());
return true;
dest
에서부터 parent
배열을 활용하여 이전 경로를 하나하나씩 탐색해가며 경로를 path
벡터에 저장한다. 저장 후, path
배열을 reverse
를 활용하여 돌려준다.
for (int p = 0;p < path.size() - 1;p++) {
...
cout << warrior_move_cnt << " " << stone_chk << " " << died_count << "\\n";
}
그리고 해당 경로 벡터를 활용하여, 아래의 절차들을 반복해준다.
전체 코드
#include <iostream>
#include <queue>
#include <vector>
#include <tuple>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
typedef struct {
int x, y;
}_loc;
typedef struct {
int x, y;
bool died;
bool stone;
}_war;
int n, m;
int start[2];
int dest[2];
_war warrior[301];
int arr[51][51];
vector<_loc> path;
void input() {
cin >> n >> m;
cin >> start[0] >> start[1] >> dest[0] >> dest[1];
for (int i = 0;i < m;i++) {
cin >> warrior[i].x >> warrior[i].y;
warrior[i].died = false;
warrior[i].stone = false;
}
for (int i = 0;i < n;i++) {
for (int j = 0;j < n;j++) {
cin >> arr[i][j];
}
}
}
int visited[51][51];
int parent[51][51][2];
bool get_path() {
queue<pair<int, int>> que;
que.push({ start[0],start[1] });
visited[start[0]][start[1]] = 1;
int dx[4] = { -1,1,0,0 };
int dy[4] = { 0,0,-1,1 };
bool found = false;
int cx, cy;
while (!que.empty()) {
tie(cx, cy) = que.front();
que.pop();
if (cx == dest[0] && cy == dest[1]) {
found = true;
break;
}
for (int i = 0;i < 4;i++) {
int nx, ny;
nx = cx + dx[i], ny = cy + dy[i];
if (nx < 0 || ny < 0 || nx >= n || ny >= n) continue;
if (visited[nx][ny] || arr[nx][ny]) continue;
visited[nx][ny] = 1;
parent[nx][ny][0] = cx, parent[nx][ny][1] = cy;
que.push({ nx,ny });
}
}
if (!found) return false;
cx = dest[0], cy = dest[1];
while (!(cx == start[0] && cy == start[1])) {
_loc tmp;
tmp.x = cx, tmp.y = cy;
path.push_back(tmp);
cx = parent[tmp.x][tmp.y][0], cy = parent[tmp.x][tmp.y][1];
}
reverse(path.begin(), path.end());
return true;
}
int marr[51][51];
int sight_arr[51][51];
void set_war_back(int x, int y, int mx, int my, int d,bool real_flag) {
int wloc;
int cnt = 1;
int s, e;
if (d == 0) {
wloc = my - y;
for (int i = x - 1;i >= 0;i--) {
if (wloc > 0) {
s = y - cnt < 0 ? 0 : y - cnt;
e = y;
}
else if (wloc < 0) {
s = y;
e = y + cnt >= n ? n - 1 : y + cnt;
}
else {
s = y, e = y;
}
for (int j = s;j <= e;j++) {
marr[i][j] = -1;
if (real_flag) sight_arr[i][j] = -1;
}
cnt++;
}
}
else if (d == 1) {
wloc = my - y;
for (int i = x + 1;i<n;i++) {
if (wloc > 0) {
s = y - cnt < 0 ? 0 : y - cnt;
e = y;
}
else if (wloc < 0) {
s = y;
e = y + cnt >= n ? n - 1 : y + cnt;
}
else {
s = y, e = y;
}
for (int j = s;j <= e;j++) {
marr[i][j] = -1;
if (real_flag) sight_arr[i][j] = -1;
}
cnt++;
}
}
else if (d == 2) {
wloc = mx - x;
for (int i = y - 1;i >= 0;i--) {
if (wloc > 0) {
s = x - cnt < 0 ? 0 : x - cnt;
e = x;
}
else if (wloc < 0) {
s = x;
e = x + cnt >= n ? n - 1 : x + cnt;
}
else {
s = x, e = x;
}
for (int j = s;j <= e;j++) {
marr[j][i] = -1;
if (real_flag) sight_arr[j][i] = -1;
}
cnt++;
}
}
else {
wloc = mx - x;
for (int i = y + 1;i < n;i++) {
if (wloc > 0) {
s = x - cnt < 0 ? 0 : x - cnt;
e = x;
}
else if (wloc < 0) {
s = x;
e = x + cnt >= n ? n - 1 : x + cnt;
}
else {
s = x, e = x;
}
for (int j = s;j <= e;j++) {
marr[j][i] = -1;
if (real_flag) sight_arr[j][i] = -1;
}
cnt++;
}
}
}
int medu_sight(int d,int x,int y,bool real_flag) { // 상,하,좌,우
int dx[4] = { -1,1,0,0 };
int dy[4] = { 0,0,-1,1 };
memset(marr, 0, sizeof(marr));
// 1. 병사들 위치 체크
for (int i = 0;i < m;i++) {
if (warrior[i].died) continue;
marr[warrior[i].x][warrior[i].y]++;
}
// 메두사 시야 설정을 위한 변수
int cnt = 1;
int s, e;
int ret = 0;
if (d == 0) {
for (int i = x - 1;i >= 0;i--) {
s = y - cnt < 0 ? 0 : y - cnt;
e = y + cnt >= n ? n - 1 : y + cnt;
for (int j = s;j <= e;j++) {
if (real_flag && sight_arr[i][j] != -1) sight_arr[i][j] = 1;
if (marr[i][j] > 0) { // 병사가 있다면
set_war_back(i, j, x, y, d,real_flag);
ret += marr[i][j];
marr[i][j] = 999;
}
}
cnt++;
}
}
else if (d == 1) {
for (int i = x + 1;i < n;i++) {
s = y - cnt < 0 ? 0 : y - cnt;
e = y + cnt >= n ? n - 1 : y + cnt;
for (int j = s;j <= e;j++) {
if (real_flag && sight_arr[i][j] != -1) sight_arr[i][j] = 1;
if (marr[i][j] > 0) { // 병사가 있다면
set_war_back(i, j, x, y, d,real_flag);
ret += marr[i][j];
marr[i][j] = 999;
}
}
cnt++;
}
}
else if (d == 2) {
for (int i = y - 1;i >= 0;i--) {
s = x - cnt < 0 ? 0 : x - cnt;
e = x + cnt >= n ? n - 1 : x + cnt;
for (int j = s;j <= e;j++) {
if (real_flag && sight_arr[j][i] != -1) sight_arr[j][i] = 1;
if (marr[j][i] > 0) { // 병사가 있다면
set_war_back(j, i, x, y, d,real_flag);
ret += marr[j][i];
marr[j][i] = 999;
}
}
cnt++;
}
}
else {
for (int i = y + 1;i < n;i++) {
s = x - cnt < 0 ? 0 : x - cnt;
e = x + cnt >= n ? n - 1 : x + cnt;
for (int j = s;j <= e;j++) {
if (real_flag && sight_arr[j][i] != -1) sight_arr[j][i] = 1;
if (marr[j][i] > 0) { // 병사가 있다면
set_war_back(j, i, x, y, d,real_flag);
ret += marr[j][i];
marr[j][i] = 999;
}
}
cnt++;
}
}
if (real_flag) {
for (int i = 0;i < m;i++) {
if (marr[warrior[i].x][warrior[i].y] == 999) {
warrior[i].stone = true;
}
}
}
return ret;
}
int warrior_move(int cur, int mx,int my, bool is_first) {
int dx[4], dy[4];
if (is_first) {
dx[0] = -1, dx[1] = 1, dx[2] = 0, dx[3] = 0;
dy[0] = 0, dy[1] = 0, dy[2] = -1, dy[3] = 1;
}
else {
dx[2] = -1, dx[3] = 1, dx[0] = 0, dx[1] = 0;
dy[2] = 0, dy[3] = 0, dy[0] = -1, dy[1] = 1;
}
int cx, cy, nx, ny;
cx = warrior[cur].x, cy = warrior[cur].y;
int cur_distance = abs(mx - cx) + abs(my - cy);
int next_distance = 987654321;
int nd = -1;
for (int d = 0;d < 4;d++) {
int tmp_distance;
nx = cx + dx[d], ny = cy + dy[d];
if (nx < 0 || ny < 0 || nx >= n || ny >= n) continue;
if (sight_arr[nx][ny] == 1) continue;
tmp_distance = abs(mx - nx) + abs(my - ny);
if (tmp_distance >= cur_distance || tmp_distance >= next_distance) continue;
next_distance = tmp_distance;
nd = d;
}
if (nd == -1) return 0;
warrior[cur].x += dx[nd], warrior[cur].y += dy[nd];
return 1;
}
int solve() {
if (!get_path()) return -1;
for (int p = 0;p < path.size() - 1;p++) {
// 모든 전사가 이동한 거리의 합 , 메두사로 인해 돌이 된 전사의 수 , 메두사를 공격한 전사의 수
// 이동한 위치에 전사가 있으면 죽이기
for (int i = 0;i < m;i++) {
if (warrior[i].x == path[p].x && warrior[i].y == path[p].y && warrior[i].died == false) {
warrior[i].died = true;
}
}
// 1. 메두사의 시선
// 1-1. 메두사가 볼 시선을 정함
int d = -1; int stone_chk = 0;
for (int i = 0;i < 4;i++) {
int cur_val = medu_sight(i, path[p].x, path[p].y,false);
if (cur_val > stone_chk) {
d = i, stone_chk = cur_val;
}
}
// 2. 메두사가 병사들은 돌로 만듬
memset(sight_arr, 0, sizeof(sight_arr));
medu_sight(d, path[p].x, path[p].y,true);
// 3. 전사들의 이동
int warrior_move_cnt = 0;
for (int i = 0;i < m;i++) {
if (warrior[i].died || warrior[i].stone) continue;
warrior_move_cnt += warrior_move(i, path[p].x, path[p].y,true);
warrior_move_cnt += warrior_move(i, path[p].x, path[p].y,false);
}
// 4. 전사들의 공격 + 돌로 변한 전사들 풀어주기
int died_count = 0;
for (int i = 0;i < m;i++) {
if (warrior[i].x == path[p].x && warrior[i].y == path[p].y && !warrior[i].died && !warrior[i].stone) {
warrior[i].died = true;
died_count++;
}
warrior[i].stone = false;
}
cout << warrior_move_cnt << " " << stone_chk << " " << died_count << "\\n";
}
return 0;
}
int main() {
ios::sync_with_stdio(0);
cin.tie(0);
input();
cout << solve();
}