알고리즘

[프로그래머스] 리코쳇 로봇 파이썬 풀이

최슬슬 2024. 9. 22. 21:56

※ 사용언어 : 파이썬 ※

 

▼ 문제 링크 ▼

https://school.programmers.co.kr/learn/courses/30/lessons/169199

 

프로그래머스

코드 중심의 개발자 채용. 스택 기반의 포지션 매칭. 프로그래머스의 개발자 맞춤형 프로필을 등록하고, 나와 기술 궁합이 잘 맞는 기업들을 매칭 받으세요.

programmers.co.kr

 


 

문제 접근

◇ 전형적인 BFS(DFS) 문제로 로봇은 격자 판에서 장애물이나 게임판의 가장자리(벽)에 부딪힐 때까지 미끄러져 이동하는 점에서 약간의 변형이 들어가 있다고 볼 수 있습니다.

 

◇ 로봇이 벽이나 장애물에 부딪힐 때 까지 이동시키는 것이 해당 문제의 핵심 포인트입니다.

 

 


 

문제 풀이 (With Python)

  풀이에 앞서, 기본적인 BFS 로직을 알고 있다는 가정 하에 서술되었습니다. BFS를 잘 모르시는 분들은 BFS를 공부한 뒤에 해당 문제를 푸는 것을 추천드립니다. (BFS 기본 로직 문제)

from collections import deque

dx = [-1,1,0,0]
dy = [0,0,-1,1]

def solution(board):
    length_row = len(board)
    length_col = len(board[0])
    start_x, start_y = 0, 0
    goal_x, goal_y = 0,0
    for i in range(length_row):
        for j in range(length_col):
            if board[i][j] == "R":
                start_x = i
                start_y = j
            elif board[i][j] == "G":
                goal_x = i
                goal_y = j

    que = deque()
    visited = [[ False for _ in range(length_col)] for _ in range(length_row)]
    que.append((start_x, start_y, 0))
    visited[start_x][start_y] = True
    
    while que:
        cx, cy, cnt = que.popleft() 
        if cx == goal_x and cy == goal_y:
            return cnt
            
        for i in range(4):
            nx = cx
            ny = cy
            while 0 <= nx+dx[i] < length_row and 0 <= ny+dy[i] < length_col and board[nx+dx[i]][ny+dy[i]] != "D":
                nx += dx[i]
                ny += dy[i]
            if not visited[nx][ny]:
                visited[nx][ny] = True
                que.append((nx,ny,cnt+1))
                
    return -1

 

◇ board를 순회하면서 시작 지점인 "R"과 목표지점인 "G"를 찾습니다.

 

◇ que에 시작 지점의 좌표값과 함께 로봇의 몇 번째 이동임을 알려주는 cnt를 넣어줍니다.

 

◇ while문을 통해 que에 들어간 좌표값과 cnt값을 꺼냅니다. 이때 들어간 좌표값이 목표지점인 goal_xgoal_y의 값과 같다면 로봇의 이동값인 cnt를 return 합니다.

 

while문 안의 for문을 통해 로봇은 아래, 위, 왼쪽, 오른쪽 순으로 이동하게 됩니다. 각 방향에 대해 어디까지 이동할 수 있는지를 파악하기 전, nxnycxcy값을 대입해서 nxny를 증가시키며 cxcy의 값을 보존합니다. 그 이유는 현재 로봇의 위치인 "cx cy"를 기준으로 상하좌우로 로봇을 움직이기 때문입니다. 만약 로봇이 시작점인 "R"처럼, 아래와 왼쪽으로 갈 수 있을 때 아래로 보낼 수 있을 때까지 보낸 값을 que에 넣고 그다음 왼쪽으로도 어디까지 갈 수 있는지 "cx cy"값으로 판단해야 합니다.

 

◇ while문을 통해 로봇이 장애물인 "D"에 부딪히거나 벽에 닿을 때까지 nxny의 값을 증가시켜 줍니다.

 

nxny의 값의 증가가 끝나면(장애물에 부딪히거나 벽에 닿음) 해당 nx, ny를 방문했는지 파악한 뒤 방문하지 않는 경우 que에 넣은 뒤 로봇이 움직인 경로로 추가되었기에 cnt를 +1 해줍니다.