새소식

PS/백준

[백준/1445] 일요일 아침의 데이트 (Python)

  • -

Problem : https://www.acmicpc.net/problem/1445

 

1445번: 일요일 아침의 데이트

첫째 줄에 숲의 세로 크기 N과 가로 크기 M이 주어진다. N과 M은 3보다 크거나 같고, 50보다 작거나 같은 자연수이다. 둘째 줄부터 숲의 지도가 주어진다. 숲의 지도는 S, F, g, . 만으로 이루어져 있

www.acmicpc.net

 

Difficulty : Gold 2

 

Status : Solved

 

Time : 00:16:07

 


 

문제 설명

 

더보기

 

일요일 아침에 형택이는 Maroon5의 Sunday Morning이란 노래를 들으면서 여자친구와의 로맨틱한 여행을 떠나기로 했다. 형택이는 이것저것 환상에 빠져있다가, 계획을 세우는데 실패했다. 따라서, 주위에 있는 숲을 같이 탐험하기로 했다.

깊은 숲속에는 정말 아름다운 꽃이 하나있다. 형택이는 여자친구의 마음을 감동시키기 위해서, 꽃을 보여주면서 자신의 마음을 전해주려고 급하게 계획했다.

불행하게도, 사람들이 숲에다 쓰레기를 버려서 형택이의 계획은 정말 망가지기 직전이다.

형택이는 그동안 여자친구와 사귀면서 2가지 깨달은 것이 있는데, 한 가지는 쓰레기를 통과해서 지나가는 것을 정말 싫어하는 것이고, 쓰레기를 따라 옆을 지나가는 것도 정말 불편하게 느낀다는 것이다.

형택이는 방금 쓰레기가 어디에있는지 조사를 마쳤다. 입력으로 숲의 지도가 주어진다. S는 형택이와 여자친구의 데이트 시작장소를  나타내고, F는 꽃이 있는 위치를 나타내고, g는 쓰레기가 있는 위치를 나타낸다. 그리고 .은 아무것도 없는 깨끗한 칸이다.

형택이의 목표는 S에서 F까지 가는데, 쓰레기로 차있는 칸을 되도록이면 적게 지나가는 것이다. 형택이와 여자친구는 한 번에 한 칸 움직일 수 있다. 가로 or 세로로 한 칸 움직일 수 있다. 만약 되도록 적게 지나가는 경우의 수가 여러개라면, 쓰레기 옆을 지나가는 칸의 개수를 최소로 해서 지나려고 한다. 만약 어떤 칸이 비어있는데, 인접한 칸에 쓰레기가 있으면 쓰레기 옆을 지나는 것이다. 그리고, S와 F는 세지 않는다.

 

 

입력 및 출력

 

더보기

입력

 

첫째 줄에 숲의 세로 크기 N과 가로 크기 M이 주어진다. N과 M은 3보다 크거나 같고, 50보다 작거나 같은 자연수이다. 둘째 줄부터 숲의 지도가 주어진다. 숲의 지도는 S, F, g, . 만으로 이루어져 있다. S는 반드시 모서리에 위치해 있고, F는 모서리에 위치해있지 않다. 그리고 S와 F는 반드시 하나만 주어진다.

 

출력

 

첫째 줄에 형택이와 여자친구가 가장 최적의 방법으로 숲을 지났을 때, 지나가는 쓰레기의 최소 개수를 출력하고, 공백으로 구분 한 후에 쓰레기 옆을 지나가는 칸의 개수를 출력한다.

 

입력 예시

 

6 6
......
g..F..
......
..g...
......
...S.g

 

출력 예시

 

0 0

 

 


 

풀이

 

엥.. 다익스트라..? 일반적인 경로 탐색으로 풀이했는데, 경로 탐색을 진행하되 힙을 사용하면(즉 다익스트라스럽게 경로를 탐색하면) 더 쉽다. 쓰레기를 지나는 경우, 쓰레기와 인접한 칸을 지나는 경우를 우선도로 두어 힙 자료구조를 사용하고, 이 힙 구조를 사용하여 탐색을 실시하면 된다. 이외에는 별다를 게 없던 무난한 문제.

 

다익스트라가 아닌, 일반적인 BFS로 풀이하여도 풀 수 있다.

 

풀이 코드

from collections import deque
import sys
input = sys.stdin.readline
dx = [0, 0, -1, 1]
dy = [-1, 1, 0, 0]
MAX = float('inf')

N, M = map(int, input().split())
map_list = [input().strip() for _ in range(N)]

garbage = set()
adj_garbage = set()

for i in range(N) :
  for j in range(M) :
    if map_list[i][j] == 'S' :
      sx, sy = j, i
    elif map_list[i][j] == 'F' :
      fx, fy = j, i
    elif map_list[i][j] == 'g' :
      garbage.add((j, i))
      for k in range(4) :
        ai, aj = i + dy[k], j + dx[k]
        if -1 < ai < N and -1 < aj < M :
          adj_garbage.add((aj, ai))

adj_garbage.discard((sx, sy))
adj_garbage.discard((fx, fy))
adj_garbage -= garbage

visited = [[(MAX, MAX) for _ in range(M)] for _ in range(N)]
visited[0][0] = (0, 0)
q = deque([(sx, sy, 0, 0)])

while q :
  x, y, g, adj_g = q.popleft()
  if (x, y) == (fx, fy) :
    continue
  for k in range(4) :
    ax, ay = x + dx[k], y + dy[k]
    ag = g + 1 if (ax, ay) in garbage else g
    a_adj_g = adj_g + 1 if (ax, ay) in adj_garbage else adj_g
    if -1 < ax < M and -1 < ay < N and visited[ay][ax] > (ag, a_adj_g) :
      visited[ay][ax] = (ag, a_adj_g)
      q.append((ax, ay, ag, a_adj_g))

print(*visited[fy][fx])

풀이 완료!

Contents

포스팅 주소를 복사했습니다

이 글이 도움이 되었다면 공감 부탁드립니다.