코딩 기록 저장소

[프로그래머스/Python] 리코쳇 로봇 본문

프로그래머스/Lv.2

[프로그래머스/Python] 리코쳇 로봇

KimNang 2024. 3. 11. 23:12

문제 정보

제목 : 리코쳇 로봇

난이도 : Lv.2

사용 언어 : Python

문제 링크

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

 

프로그래머스

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

programmers.co.kr

 

문제 설명

리코쳇 로봇이라는 보드게임이 있습니다.

이 보드게임은 격자모양 게임판 위에서 말을 움직이는 게임으로, 시작 위치에서 목표 위치까지 최소 몇 번만에 도달할 수 있는지 말하는 게임입니다.

이 게임에서 말의 움직임은 상, 하, 좌, 우 4방향 중 하나를 선택해서 게임판 위의 장애물이나 맨 끝에 부딪힐 때까지 미끄러져 이동하는 것을 한 번의 이동으로 칩니다.

다음은 보드게임판을 나타낸 예시입니다.

...D..R
.D.G...
....D.D
D....D.
..D....

여기서 "."은 빈 공간을, "R"은 로봇의 처음 위치를, "D"는 장애물의 위치를, "G"는 목표지점을 나타냅니다.
위 예시에서는 "R" 위치에서 아래, 왼쪽, 위, 왼쪽, 아래, 오른쪽, 위 순서로 움직이면 7번 만에 "G" 위치에 멈춰 설 수 있으며, 이것이 최소 움직임 중 하나입니다.

게임판의 상태를 나타내는 문자열 배열 board가 주어졌을 때, 말이 목표위치에 도달하는데 최소 몇 번 이동해야 하는지 return 하는 solution함수를 완성하세요. 만약 목표위치에 도달할 수 없다면 -1을 return 해주세요.

 

제한 사항

  • 3 ≤ board의 길이 ≤ 100
    • 3 ≤ board의 원소의 길이 ≤ 100
    • board의 원소의 길이는 모두 동일합니다.
    • 문자열은 ".", "D", "R", "G"로만 구성되어 있으며 각각 빈 공간, 장애물, 로봇의 처음 위치, 목표 지점을 나타냅니다.
    • "R"과 "G"는 한 번씩 등장합니다.

입출력 예

board result
["...D..R", ".D.G...", "....D.D", "D....D.", "..D...."] 7
[".D.R", "....", ".G..", "...D"] -1

 

입출력 예 설명

입출력 예 #1

  • 문제 설명의 예시와 같습니다.

입출력 예 #2

.D.R
....
.G..
...D
  • "R" 위치에 있는 말을 어떻게 움직여도 "G" 에 도달시킬 수 없습니다.
  • 따라서 -1을 return 합니다.

나의 풀이

BFS를 이용해 문제를 해결했습니다. queue에 요소가 있는 동안 반복하는 while루프와 4방향을 순회하는 for루프를 이용하는데 이때 while루프를 이용해 맵의 끝이나 장애물이 있을때 까지 반복해서 값을 더해줍니다. 내부의 while문에서 빠져나오면 조건에 맞지 않다는 것이므로 현재 인덱스는 유효한 범위를 벗어나게 되었습니다. 말의 위치를 올바르게 저장하기 위해 -를 합니다. 현재 위치가 방문하지 않은 곳이면 방문한 리스트에 값을 저장하고 queue에 위치를 저장하고 다시 반복을 수행합니다.

현재 위치가 목적지라면 현재 방문한 리스트의 값에서 -1를 해서 return하고, 모든 반복이 끝날때까지 못찾았다면 목적지에 도달할 수 없다는 뜻이므로 -1을 return합니다.

 

코드

from collections import deque

def solution(board):
    answer = 0
    queue = deque()
    for i in range(len(board)):
        board[i] = list(board[i])
        if("R" in board[i]) :
            queue.append( (i,board[i].index("R")) )
    answer = BFS(board,queue)
    return answer


def BFS(board, queue) :
    lenY = len(board)
    lenX = len(board[0])
    visitedBoard = [[0 for _ in range(lenX)] for _ in range(lenY) ] # 방문한 위치 체크함
    visitedBoard[queue[0][0]][queue[0][1]] = 1
    
    dY = [1,0,-1,0]
    dX = [0,1,0,-1]
    
    while(queue) :
        curY, curX = queue.popleft()
        
        if(board[curY][curX] == "G"): # 목적지라면 -1해서 리턴함
            return visitedBoard[curY][curX]-1
        
        for i in range(4) :
            nextY, nextX = curY+dY[i], curX+dX[i] # 다음 방향 설정
            # 다음위치가 범위 안이고 장애물이 아니면 다음 위치 값 더함
            while(0 <= nextY < lenY and 0 <= nextX < lenX and board[nextY][nextX] != "D"):
                    nextY += dY[i]
                    nextX += dX[i]
                    
            # while문을 벗어났기 때문에 이전 위치를 만들기 위해 -해줌        
            nextY -= dY[i]
            nextX -= dX[i]
            if(visitedBoard[nextY][nextX] == 0): # 방문 안한 곳이면 queue에 추가
                visitedBoard[nextY][nextX] = visitedBoard[curY][curX]+1
                queue.append((nextY,nextX))
    return -1