알고리즘

BOJ 23290 - 마법사 상어와 복제(C++)

장중앙 2021. 11. 27. 17:55

https://www.acmicpc.net/problem/23290

 

23290번: 마법사 상어와 복제

첫째 줄에 물고기의 수 M, 상어가 마법을 연습한 횟수 S가 주어진다. 둘째 줄부터 M개의 줄에는 물고기의 정보 fx, fy, d가 주어진다. (fx, fy)는 물고기의 위치를 의미하고, d는 방향을 의미한다. 방향

www.acmicpc.net

 

풀이

한 위치에 물고기가 여러 마리일 수 도 있기 떄문에 vector<int> fish[][]로 저장, 값은 물고기의 방향성

 

상어가 물고기를 먹었다면 냄새를 남기는 이차원 배열 smell[][]을 이용

smell[y][x]= 현재 연습 시점의 index을 남기고, 물고기가 이동할 때 이동 가능한지의 여부를 확인 할때,

현재 연습 시점 index - smell[y][x] 가 2이하라면 냄새가 사라지지 않았다고 판별 -> smell배열에서 냄새를 남기고 2초뒈에 삭제를 할 필요 X

 

상어의 이동 방법은 먹을 물고기의 양이 같다면, 사전순으로 앞서는 방법을 우선적으로 선택하기 때문에

우선 순위인 상, 좌, 하, 우에 맞춰 dy[-1, 0, 1, 0]={}, dx[]={0, -1, 0, 1}로 설정

 

연습 횟수 S번 만큼 1~5를 반복

1. 현재 상태의 물고기 맵 fish[][]를 복사하여 copy[][]에 저장 

2. 모든 물고기 이동

    - 상어가 있는 칸, 물고기 냄새가 있는 칸, 격자의 범위를 벗어난 다는 위치로는 이동 불가

    - 물고기의 냄새가 있는 지 판별법(냄새는 남긴 시간으로 부터 2연습 뒤 사라짐) 

        - smell[y][x]의 값이 현재 시간 time - smell[y][x]의 값 <=2

    - 이동이 가능할 때 까지 반시계방향 45도 회전 (dir = (8+dir-1) % 8)

3. 상어 이동 시뮬레이션

    - 현재 위치에서 상하좌우 인접한 칸으로 3번 이동(dfs)

    - 물고기를 가장 많이 먹을 수 있는 경로, 먹을 수 있는 물고기의 수가 같다면 사전순으로 앞서는 경로를 선택(우선순위 : 상, 좌, 하, 우)

4. 3에서 정해진 경로로 상어 이동

    - 이동 중 현재 위치에 물고기가 있다면 해당위치의 물고기를 전부 삭제

    - 삭제되는 위치에는 물고기의 냄새를 남김

        - 물고리의 냄새는 현재 연습 횟수 index로 표시(smell[y][x]=time)

5. 1에서 실행한 물고기 이동전의 배열을 현재 fish[][]에 붙여넣기

 

 

#include <iostream>
#include <vector>
using namespace std;
#define MAX 5

int M, S;
int shark_y, shark_x;
int dy[] = { 0,-1,-1,-1,0,1,1,1 };
int dx[] = { -1,-1,0,1,1,1,0,-1 };
int s_dy[] = { 0,-1,0,1,0 };
int s_dx[] = { 0,0,-1,0,1 };
int smell[MAX][MAX];
vector<int> fish[MAX][MAX];
vector<int> copy_fish[MAX][MAX];
vector<int> shark_move;
int max_eat;


void copy_map(vector<int> a[][MAX], vector<int> b[][MAX]) {
    for (int i = 1; i < MAX; i++) {
        for (int j = 1; j < MAX; j++) {
            b[i][j] = a[i][j];
        }
    }
}

void move_fish(int time) {
    vector<int> new_fish[MAX][MAX];
	// 이차원 배열 탐색, 현재 물고기 있으면 물고기마다 위치 이동 및 이동 방향 재설정
    for (int i = 1; i < MAX; i++) {
        for (int j = 1; j < MAX; j++) {
            for (int k = 0; k < fish[i][j].size(); k++) {
                int y = i;
                int x = j;
                int dir = fish[i][j][k];
                      
                int ndir, ny, nx;
                bool flag = false;
                for (int i = 0; i < 8; i++) {
                    ndir = (8 + dir - i) % 8;
                    ny = y + dy[ndir];
                    nx = x + dx[ndir];
                    if (ny < 1 || nx < 1 || ny >= MAX || nx >= MAX)continue;
                    if (ny == shark_y && nx == shark_x)continue;
                    if (smell[ny][nx] != 0 && time-smell[ny][nx] <= 2)continue;
                    flag = true;
                    new_fish[ny][nx].push_back(ndir);
                    break;
                }
                if (!flag) {
                    new_fish[y][x].push_back(dir);
                }
            }
        }
    }
    copy_map(new_fish, fish);
}

int shark_simulation(vector<int> route) {
    bool visit[MAX][MAX] = { false, };
    int res = 0;
    int s_y = shark_y;
    int s_x = shark_x;
    for (auto dir : route) {
        s_y += s_dy[dir];
        s_x += s_dx[dir];
        if (s_y < 1 || s_x < 1 || s_y >= MAX || s_x >= MAX)return -1;
        if(!visit[s_y][s_x]){
            visit[s_y][s_x] = true;
            res += fish[s_y][s_x].size();
        }
    }
    return res;
}
// 상어 이동 시뮬레이션, 어디로 가는 것이 물고기를 최대로 먹을지
void search_best_move(vector<int> route) {
    if (route.size() == 3) {
    	// 3번 이동 했다면 물고기를 몇 마리 먹을 수 있는지 확인
        int eat = shark_simulation(route);
        if (eat > max_eat) {
            max_eat = eat;
            shark_move = route;
        }
        return;
    }
    for (int i = 1; i <= 4; i++) {
        route.push_back(i);
        search_best_move(route);
        route.pop_back();
    }
}

// 이동방향이 결정되었다면 상어 이동, 이동 중 물고기를 먹었다면 해당 위치에 냄새를 남김
void move_shark(int time) {
    for (auto dir : shark_move) {
        shark_y += s_dy[dir];
        shark_x += s_dx[dir];
        if (fish[shark_y][shark_x].size() != 0) {
            fish[shark_y][shark_x].clear();
            smell[shark_y][shark_x] = time;
        }
    }
}

void paste_fish() {
    for (int i = 1; i < MAX; i++) {
        for (int j = 1; j < MAX; j++) {
            for(auto dir:copy_fish[i][j]){
                fish[i][j].push_back(dir);
            }
        }
    }
}

int main() {
    ios_base::sync_with_stdio(false);
	cin.tie(0);
    
    cin >> M >> S;
    int y, x, dir;
    for (int i = 1; i <= M; i++) {
        cin >> y >> x >> dir;
        fish[y][x].push_back(dir-1);
    }
    cin >> shark_y >> shark_x;
    
    for (int i = 1; i <= S; i++) {
    	//현재 물고기 위치 저장
        copy_map(fish, copy_fish);
        move_fish(i);
        // 상어 이동 시뮬레이션을 위한 설정, 상어 이동
        max_eat = -1;
        vector<int> temp;
        search_best_move(temp);
        move_shark(i);
        // 저장된 물고기 위치 복제
        paste_fish();
    }
    
	// 현재 남아 있는 물고기 수 
    int res = 0;
    for (int i = 1; i < MAX; i++) {
        for (int j = 1; j < MAX; j++) {
            res += fish[i][j].size();
        }
    }
    cout << res;
    return 0;
}