알고리즘/문제풀이

[백준 17086번] 아기 상어 2

Ohnim · 오님 2024. 5. 26. 02:46

문제 링크

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

 

문제 요약

임의의 점 \((i, j)\)의 안전거리는 \((i, j)\)에서 시작해서 아기 상어가 있는 곳까지의 최소 거리를 의미한다. 여기서 거리는 인접한 8방향(상하좌우 및 대각선)으로 이동한 칸 수로 계산한다.

 

격자판 위의 모든 점들 중에서 안전거리가 최대인 값을 찾아 출력하는 것이 이 문제의 목표이다.

 

문제 풀이

기본적인 풀이

너비 우선 탐색(이하 BFS)을 이용하면 임의의 점 \((i, j)\)에서 가장 가까운 아기 상어까지의 거리를 계산할 수 있다. 그리고 N과 M의 크기가 작으므로 모든 점들에 대해서 BFS를 한 번씩 돌려도 제한시간 내에 충분히 답을 구할 수 있다. 아래 더보기 버튼을 누르면 이 방법으로 푼 코드를 볼 수 있다.

 

더보기
더보기
#include <stdio.h>
#include <string.h>
#include <queue>
#include <algorithm>

using namespace std;

const int dx[] = {-1, -1, -1, 0, 1, 1, 1, 0};
const int dy[] = {-1, 0, 1, 1, 1, 0, -1, -1};
int n, m, A[51][51], D[51][51];

bool isPossible(int x, int y) {
    if (x == -1 || x == n || y == -1 || y == m) {
        return false;
    }
    
    return true;
}

int bfs(pair<int, int> src) {
    queue<pair<int, int>> q;
    memset(D, -1, sizeof(D));
    
    q.push(src);
    D[src.first][src.second] = 0;
    
    while (!q.empty()) {
        int x = q.front().first;
        int y = q.front().second;
        q.pop();
        
        if (A[x][y] == 1) {
            return D[x][y];
        }
        
        for (int i = 0 ; i < 8 ; ++i) {
            int nx = x + dx[i], ny = y + dy[i];
            
            if (isPossible(nx, ny) && D[nx][ny] == -1) {
                q.push({ nx, ny });
                D[nx][ny] = D[x][y] + 1;
            }
        }
    }
    
    return 0;
}

int main() {
    scanf("%d%d", &n, &m);
    
    for (int i = 0 ; i < n ; ++i) {
        for (int j = 0 ; j < m ; ++j) {
            scanf("%d", &A[i][j]);
        }
    }
    
    int ans = 0;
    
    for (int i = 0 ; i < n ; ++i) {
        for (int j = 0 ; j < m ; ++j) {
            if (A[i][j] == 0) {
                ans = max(ans, bfs({ i, j }));
            }
        }
    }
    
    printf("%d\n", ans);
}

 

조금 더 빠르게 풀어보기

격자판 위의 모든 점에 대해서 각각 BFS를 돌리는 것이 비효율적이라고 생각할 수 있다. 만약 문제에서 주어지는 N과 M의 최대 크기가 지금보다 더 컸다면 위 방법으로는 이 문제를 풀 수 없었을 것이다. 조금 더 빠르게 풀 수 있는 방법은 없을까?

 

BFS를 돌리기 위해서 암시적으로 모델링 한 그래프가 무향 그래프라는 점을 생각해 보자. 그러면 각 점에서 아기 상어까지의 최단 거리와 아기 상어에서 각 점까지의 최단 거리가 같다는 것을 알 수 있다. 따라서 아기 상어가 있는 모든 지점들을 시작점으로 잡고 BFS를 돌리면 BFS 한 번에 모든 지점들의 안전거리를 구할 수 있다. 아래 더보기 버튼을 누르면 더 빠르게 문제를 해결하는 코드를 볼 수 있다.

 

더보기
더보기
#include <stdio.h>
#include <string.h>
#include <queue>
#include <algorithm>

using namespace std;

const int dx[] = {-1, -1, -1, 0, 1, 1, 1, 0};
const int dy[] = {-1, 0, 1, 1, 1, 0, -1, -1};
int n, m, A[51][51], D[51][51];

bool isPossible(int x, int y) {
    if (x == -1 || x == n || y == -1 || y == m) {
        return false;
    }
    
    return true;
}

int main() {
    scanf("%d%d", &n, &m);
    
    for (int i = 0 ; i < n ; ++i) {
        for (int j = 0 ; j < m ; ++j) {
            scanf("%d", &A[i][j]);
        }
    }
    
    memset(D, -1, sizeof(D));
    queue<pair<int, int>> q;
    
    for (int i = 0 ; i < n ; ++i) {
        for (int j = 0 ; j < m ; ++j) {
            if (A[i][j] == 1) {
                D[i][j] = 0;
                q.push({ i, j });
            }
        }
    }
    
    while (!q.empty()) {
        int x = q.front().first;
        int y = q.front().second;
        q.pop();
        
        for (int i = 0 ; i < 8 ; ++i) {
            int nx = x + dx[i], ny = y + dy[i];
            
            if (isPossible(nx, ny) && D[nx][ny] == -1) {
                D[nx][ny] = D[x][y] + 1;
                q.push({ nx, ny });
            }
        }
    }
    
    int ans = 0;
    
    for (int i = 0 ; i < n ; ++i) {
        for (int j = 0 ; j < m ; ++j) {
            ans = max(ans, D[i][j]);
        }
    }
    
    printf("%d\n", ans);
}