[백준 2573] 빙산

코뉴·2022년 2월 8일
0

백준🍳

목록 보기
99/149
post-custom-banner

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

🥚문제


🥚입력/출력


🍳코드

PyPy3로 통과

import sys
from collections import deque
input = sys.stdin.readline

# 인자로 받은 arr 상태에서 (r, c)에 위치한 빙산을 녹이고, 
# 빙산이 녹은 뒤의 높이를 리턴하는 함수
def melt(r, c, arr):
    d_r = [0, 0, -1, 1]
    d_c = [-1, 1, 0, 0]
    cnt = 0  # 바닷물에 접하는 부분의 개수

    for i in range(4):
        n_r = r + d_r[i]
        n_c = c + d_c[i]
        if arr[n_r][n_c] == 0:
            cnt += 1

    # 녹은 뒤의 빙산 높이
    ret = arr[r][c] - cnt
    if ret < 0:
        return 0
    return ret

# bfs로 빙산의 개수를 세줌
def bfs(r, c, arr, visited):
    d_r = [0, 0, -1, 1]
    d_c = [-1, 1, 0, 0]
    q = deque([(r, c)])
    visited[r][c] = True

    while q:
        r, c = q.popleft()
        for i in range(4):
            n_r = r + d_r[i]
            n_c = c + d_c[i]

            # (n_r, n_c)가 빙산
            if arr[n_r][n_c] > 0 and not visited[n_r][n_c]:
                q.append((n_r, n_c))
                visited[n_r][n_c] = True


# 입력
N, M = map(int, input().split())
arr = [list(map(int, input().split())) for _ in range(N)]

ans = 0
while True:
    # 빙산이 분리되는지 체크
    visited = [[False]*M for _ in range(N)]
    bfs_count = 0
    for r in range(N):
        for c in range(M):
            if arr[r][c] > 0 and not visited[r][c]:
                bfs(r, c, arr, visited)
                bfs_count += 1

    # 빙산이 분리된다
    if bfs_count > 1:
        print(ans)
        break

    # 빙산이 다 녹았는지 체크
    all_melt = True
    for r in range(N):
        for c in range(M):
            if arr[r][c] > 0:
                all_melt = False
                break

    if all_melt:
        print(0)
        break

    # ice: 빙산의 위치를 저장하는 리스트
    ice = []
    for r in range(N):
        for c in range(M):
            if arr[r][c] > 0:
                ice.append((r, c))

    # 현재 arr의 빙산을 녹인 후의 상태를 저장하는 리스트 new_arr
    new_arr = [[0]*M for _ in range(N)]
    for info in ice:
        r, c = info
        new_arr[r][c] = melt(r, c, arr)

    # new_arr의 상태를 arr에 다시 저장
    arr = [row[:] for row in new_arr]

    # ans 증가
    ans += 1

🧂아이디어

구현, 탐색

빙산 녹이기: melt(r, c, arr)

  • melt(r, c, arr) 함수는 인자로 받은 arr 상태에서 (r, c)에 위치한 빙산을 녹이고, 빙산이 녹은 뒤의 높이를 리턴한다.

  • 현재 arr에 위치한 빙산들의 좌표 (r, c)에 대해 melt(r, c, arr)을 시행한 뒤, 리턴받는 값은 새로운 리스트 new_arr(r, c) 위치에 저장해야 한다.

  • 이후, arrnew_arr의 값을 갖도록 업데이트 해준다.

빙산 개수 세기: bfs(r, c, arr, visited)

  • bfs(r, c, arr, visited) 함수는 인자로 받은 arr 상태의 (r, c)에 위치한 빙산에서 시작해서 상, 하, 좌, 우 방향으로 위치한 다른 빙산들을 탐색하는 넓이 우선 탐색을 시행한다.
  • bfs를 시행한 횟수 = 빙산의 개수
  • 입력으로 빙산이 이미 분리되어 있는 상태가 주어질 수 있고, 빙산이 분리되는 가장 최초의 시간을 출력해야하므로 while문 내부에서 가장 우선적으로 빙산 개수를 세준다.
profile
코뉴의 도딩기록
post-custom-banner

0개의 댓글