[Programmers] 미로탈출

Coodori·2023년 3월 27일
0

Programmers

목록 보기
4/10

문제

문제 설명
1 x 1 크기의 칸들로 이루어진 직사각형 격자 형태의 미로에서 탈출하려고 합니다. 각 칸은 통로 또는 벽으로 구성되어 있으며, 벽으로 된 칸은 지나갈 수 없고 통로로 된 칸으로만 이동할 수 있습니다. 통로들 중 한 칸에는 미로를 빠져나가는 문이 있는데, 이 문은 레버를 당겨서만 열 수 있습니다. 레버 또한 통로들 중 한 칸에 있습니다. 따라서, 출발 지점에서 먼저 레버가 있는 칸으로 이동하여 레버를 당긴 후 미로를 빠져나가는 문이 있는 칸으로 이동하면 됩니다. 이때 아직 레버를 당기지 않았더라도 출구가 있는 칸을 지나갈 수 있습니다. 미로에서 한 칸을 이동하는데 1초가 걸린다고 할 때, 최대한 빠르게 미로를 빠져나가는데 걸리는 시간을 구하려 합니다.

미로를 나타낸 문자열 배열 maps가 매개변수로 주어질 때, 미로를 탈출하는데 필요한 최소 시간을 return 하는 solution 함수를 완성해주세요. 만약, 탈출할 수 없다면 -1을 return 해주세요.

링크 : https://school.programmers.co.kr/learn/courses/30/lessons/159993

풀이

from collections import deque
INF = int(1e9)
def solution(maps):
    answer = 0
    start_x, start_y = 0,0
    btn_x, btn_y = 0,0
    exit_x, exit_y = 0,0
    for i in range(len(maps)):
        for j in range(len(maps[0])):
            if maps[i][j] == "S":
                start_x = i
                start_y = j
            if maps[i][j] == "E":
                exit_x ,exit_y = i,j
            if maps[i][j] == "L":
                btn_x, btn_y = i, j                
    visited = bfs(start_x,start_y,maps)
    if visited[btn_x][btn_y] == INF:
        return -1
    else:
        answer += visited[btn_x][btn_y]
    visited = bfs(btn_x,btn_y,maps)
    if visited[exit_x][exit_y] == INF:
        return -1
    else:
        answer += visited[exit_x][exit_y]
    return answer

def bfs(x,y,maps):
    q = deque()
    q.append((x,y))
    visited = [[INF] * len(maps[0]) for _ in range(len(maps))]
    dx = [-1,0,1,0]
    dy = [0,1,0,-1]
    visited[x][y] = 0
    while q:
        a,b = q.popleft()
        for i in range(4):
            nx =  a + dx[i]
            ny =  b + dy[i]
            if nx < 0 or nx >= len(maps) or ny < 0 or ny >= len(maps[0]):
                continue
            if maps[nx][ny] == "X":
                continue
            if visited[nx][ny] > visited[a][b] + 1:
                q.append((nx,ny))
                visited[nx][ny] = visited[a][b] + 1
    return visited

방법

전형적인 BFS 문제 느낌이 났다.
대놓고 벽과 가까운 지점을 가야하는 문제였기 때문에 LV.2 에 속했던 것 같다.
여기서 포인트는 평소 BFS는 한번만 하는거지만
백준 트리의 지름이라는 문제와 비슷하게 start지점과 end지점을 바꿔서 두번 구현하는 것이 포인트였다.

위에서 해당 가야하는 특이점 3개를 찾아준뒤 전형적인 bfs 함수를 구현하여 해결한 것이 포인트였다.

깨달은 점

Programmers 3단계는 너무 오래걸리지만 풀었을때 쾌감이있고
2단계는 너무 아는 선에서 기출문제를 푸는 듯한 느낌이 있어서 복습하기 좋은 것 같다.
알고리즘 스터디를 매일 1~2문제씩 진행하고 있는데 꾸준하게 진행해서 감을 잃지 않게 하는 것이 중요할 것 같다.

profile
https://coodori.notion.site/0b6587977c104158be520995523b7640

0개의 댓글