Notice
Recent Posts
Recent Comments
Link
«   2025/01   »
1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31
Tags more
Archives
Today
Total
관리 메뉴

개발자 일기장.

백준 2573. 빙산 본문

취업/Algorithm.

백준 2573. 빙산

Azderica 2020. 5. 1. 23:45

2573 빙산

문제해결 사진

image

문제 알고리즘

  • bfs
  • 시뮬레이션

풀이방법

문제를 3단계로 나눠서 풀었다.

  1. 주변에 있는 개수를 체크하여 빙산을 업데이트
  2. 빙산이 두 덩어리 이상으로 분리되었는지 확인하고, 분리되었으면 종료
  3. 빙산이 없는지 체크하고 없으면 0 출력
  • 핵심 코드
#include    <iostream>
#include    <queue>
#include    <algorithm>
using namespace std;

// static
const int MAX = 300;
const pair<int, int> dir[4] = { {-1, 0}, {0, 1}, {1, 0}, {0, -1} };
// variable
int board[MAX][MAX], cpyboard[MAX][MAX];
int N, M;
int answer;                    // year

bool visited[MAX][MAX];
queue<pair<int, int>> q;

// 범위내 체크
bool isInRange(int y, int x) {
    if (y < 0 || y > N || x < 0 || x > M)
        return false;
    return true;
}

// 빙산 업데이트
void updateGlacier() {
    // origin -> cpy
    for (int i = 0; i < N; i++)
        for (int j = 0; j < M; j++)
            cpyboard[i][j] = board[i][j];

    for (int i = 0; i < N; i++) {
        for (int j = 0; j < M; j++) {
            if (board[i][j]) {
                int cnt = 0;
                for (int d = 0; d < 4; d++) {
                    int nextY = i + dir[d].first;
                    int nextX = j + dir[d].second;

                    // 범위안이고 주변에 바다인 경우
                    if (isInRange(nextY, nextX) && board[nextY][nextX] == 0) {
                        cnt++;
                    }
                }

                if (cpyboard[i][j] >= cnt)
                    cpyboard[i][j] -= cnt;
                else
                    cpyboard[i][j] = 0;
            }
        }
    }

    // cpy -> origin
    for (int i = 0; i < N; i++)
        for (int j = 0; j < M; j++)
            board[i][j] = cpyboard[i][j];
}


// 빙하가 두개로 분리되어 있으면 종료
bool isSeperated() {
    bool isOne = false;        // 한개인지 체크
    for (int i = 0; i < N; i++)
        for (int j = 0; j < M; j++)
            visited[i][j] = false;

    for (int i = 0; i < N; i++) {
        for (int j = 0; j < M; j++) {
            // 빙하가 아닌 경우
            if (board[i][j] && !visited[i][j]) {
                if (!isOne) {
                    q.push({ i, j });
                    visited[i][j] = true;
                    isOne = true;

                    while (!q.empty()) {
                        int curY = q.front().first;
                        int curX = q.front().second;
                        q.pop();

                        for (int d = 0; d < 4; d++) {
                            int nextY = curY + dir[d].first;
                            int nextX = curX + dir[d].second;

                            if (isInRange(nextY, nextX) && board[nextY][nextX] && !visited[nextY][nextX]) {
                                q.push({ nextY, nextX });
                                visited[nextY][nextX] = true;
                            }
                        }
                    }
                }
                else
                    return true;    // 두번째 덩어리가 있는 경우    
            }
        }
    }
    return false;
}

// 빙하가 없으면  true
bool isNoGlacier() {
    for (int i = 0; i < N; i++)
        for (int j = 0; j < M; j++)
            if (board[i][j])
                return false;
    return true;
}

int main() {
    // input
    cin >> N >> M;
    for (int i = 0; i < N; i++) {
        for (int j = 0; j < M; j++) {
            scanf("%d", &board[i][j]);
        }
    }

    // solution
    while (true) {
        answer++;

        updateGlacier();

        // check code
    /*    cout << answer << "\n";
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < M; j++)
                printf("%d ", board[i][j]);
            cout << "\n";
        }*/

        if (isSeperated()) {
            break;
        }
        if (isNoGlacier()) {
            answer = 0;
            break;
        }
    }

    // output
    cout << answer << "\n";
    return 0;
}

문제 후 느낀점

  • 생각한대로 바로 풀린 문제.