본문 바로가기
카테고리 없음

[백준 7569번] 토마토(시즌 2)

by 킹우현 2023. 2. 16.

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

# 가로(m), 세로(n), 높이(h)
m, n, h = map(int,input().split())

# 3차원 리스트 선언 및 초기화
area = [[[0]*m for _ in range(n)] for _ in range(h)]

# 3차원 리스트 입력받기
for i in range(h):
  for j in range(n):
    area[i][j] = list(map(int,input().split()))

# 값이 1인 인덱스(높이,세로,가로)배열
one_list = []
# 값이 0인 인덱스(높이,세로,가로)배열
zero_list = []

# 3차원 리스트에서 값이 1인 인덱스와 값이 0인 인덱스를 각 배열에 저장
for i in range(h):
  for j in range(n):
    for k in range(m):
      if area[i][j][k] == 1:
        one_list.append((i,j,k))
      elif area[i][j][k] == 0:
        zero_list.append((i,j,k))

# 만약에 처음부터 익지 않은 토마토가 없는 경우 0을 출력하고 종료
if len(zero_list) == 0:
  print(0)
  sys.exit()

# 정답(토마토가 최종적으로 다 익을 때까지 걸리는 최소 일수)
solution = 0

# BFS 함수
def bfs(one_list):

  global solution

  queue = deque([])

  # 먼저 익은 토마토(1)부터 큐에 삽입
  for i in one_list:
    if area[i[0]][i[1]][i[2]] <= 0:
      return False

    queue.append(i)

  # 큐가 비워질 때까지
  while queue:

    v = queue.popleft()

    # 상하좌우, 위, 아래 순회를 위한 값
    nx = [-1,1,0,0,0,0]
    ny = [0,0,-1,1,0,0]
    nz = [0,0,0,0,-1,1]

    # 상하좌우, 위, 아래를 탐색
    for i in range(6):
      temp_z = v[0] + nz[i]
      temp_x = v[1] + nx[i]
      temp_y = v[2] + ny[i]

      
      # 배열의 범위를 벗어나지 않는지 확인
      if temp_z >=0 and temp_z < h and temp_x >= 0 and temp_x < n and temp_y >= 0 and temp_y < m:
        # 익지 않은 토마토가 존재하는지 확인
        if area[temp_z][temp_x][temp_y] == 0:
          # 큐에 삽입
          queue.append((temp_z,temp_x,temp_y))
          # 다음 영역의 값은 현재 영역 + 1(하루가 지날 때 마다 카운팅)
          area[temp_z][temp_x][temp_y] = area[v[0]][v[1]][v[2]] + 1
          # 이동할 때 마다 최소 일수 업데이트
          solution = area[v[0]][v[1]][v[2]] + 1

# 익은 토마토(1)의 인덱스가 저장된 배열을 BFS 함수에 전달
bfs(one_list)

# 시간이 지나도 익지 못하는 토마토(0)이 존재하는 경우 -1을 출력하고 종료
for i in range(h):
  for j in range(n):
    for k in range(m):
      if area[i][j][k] == 0:
        print(-1)
        sys.exit()

# 최종적인 답은 걸린 일수이므로 -1 해주기
print(solution-1)

이번 문제는 저번에 풀었던 토마토 문제( https://woohyun-king.tistory.com/44 )를 응용한 문제이다.

 

[백준 7576번] 토마토

from collections import deque # 가로(m), 세로(n) m, n = map(int,input().split()) # 토마토가 담긴 상자(2차원 리스트) area = [] # 토마토가 존재하는 위치를 담는 리스트 exist_list = [] # 토마도가 모두 익는 최소 일수

woohyun-king.tistory.com

저번 토마토 문제와 다른 점이라고 한다면, 영역의 범위가 2차원 리스트가 아니라 3차원 리스트라는 점이다.

이것만 제외하면 BFS 알고리즘을 사용하여 각 익은 토마토로부터 인접한 토마토가 익기 시작했을 때 토마토가 다 익기 까지 걸리는 최소 일수를 구하는 문제라는 점은 동일하다 !

 

3차원 리스트를 다루는 문제라 조금 어색했지만 차근차근 조건에 맞게 구현하니 한번에 풀 수 있었다 :)