알고리즘

파이썬) 벽 부수고 이동하기 - BOJ

1일1공부실천하자 2023. 12. 10. 20:57

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

 

2206번: 벽 부수고 이동하기

N×M의 행렬로 표현되는 맵이 있다. 맵에서 0은 이동할 수 있는 곳을 나타내고, 1은 이동할 수 없는 벽이 있는 곳을 나타낸다. 당신은 (1, 1)에서 (N, M)의 위치까지 이동하려 하는데, 이때 최단 경로

www.acmicpc.net

 

from collections import deque
import sys

input = sys.stdin.readline

n, m = map(int, input().split())

arr = [list(map(int, input().rstrip())) for _ in range(n)]

visit = [[[0] * 2 for _ in range(m)] for _ in range(n)]


visit[0][0][0] = 1

dx = [0, 0, -1, 1]
dy = [-1, 1, 0, 0]
q = deque()


q.append((0, 0, 0))

condition = False

while q:
    x, y, wall = q.popleft()

    if x == n-1 and y == m-1:
        condition = True
        print(visit[x][y][wall])
        break

    for i in range(4):
        nx = x + dx[i]
        ny = y + dy[i]

        if 0 <= nx < n and 0 <= ny < m and visit[nx][ny][wall] == 0:
            if arr[nx][ny] == 0:
                q.append((nx, ny, wall))
                visit[nx][ny][wall] = visit[x][y][wall] + 1

            if wall == 0 and arr[nx][ny] == 1:
                q.append((nx, ny, 1))
                visit[nx][ny][1] = visit[x][y][wall] + 1

for i in visit:
    print(i)
if not condition:
    print(-1)

 

처음엔 단순히 큐에 담는 wall이란 변수로 내가 한번도 부수지 않았으면 0, 한번이라도 부수었다면 1을 저장해

if arr[nx][ny] == 1 and wall == 0:

으로 벽을 부수고 이동했었다.

왜냐하면 경로상에 있는 벽을 부수고 이동하는 것이 "무조건" 경로가 짧아진다고 생각했었기 때문이다.

 

그러나 벽을 부수고 이동하는 것이 최단경로에서 좀 더 멀어진다는 생각을 하지 못했었다.

따라서 벽을 부수고 이동하는 경로와 벽을 부수고 이동하지 않는 경우 둘 모두를 계산해야했다.

결국 "벽을 안부스고 [n-1][m-1]에 이동한다 => 출력, 또는 벽을 부수고 이동하여 [n-1][m-1]에 도착했다 => 출력."으로 지금 경로가 최적의 경로라는 넓이 우선탐색에 맞는 알고리즘을 선택해야했다.

 

visit은 3차원 리스트로 x,y값과 wall값을 지정해주었다.

예를들어 x = 1, y = 2 일때 visit[x][y] 는 [0,0]이 된다. 여기서 visit[x][y][0] 은 벽을 안부수고 이동했을 때의 경로, visit[x][y][1]은 벽을 부수고 이동했을 때의 경로이다.

이후 while문을 돌며 bfs알고리즘을 반복적으로 실행해주면 된다.