[백준 2253번] 점프

zeo·2021년 8월 30일
0

백준 2253번 문제 점프

문제

N(2 ≤ N ≤ 10,000)개의 돌들이 같은 간격으로 놓여 있다. 편의상 순서대로 1, 2, …, N번 돌이라고 부르자. 당신은 현재 1번 돌 위에 있는데, 이 돌들 사이에서 점프를 하면서 N번째 돌로 이동을 하려 한다. 이때 다음 조건들이 만족되어야 한다.

이동은 앞으로만 할 수 있다. 즉, 돌 번호가 증가하는 순서대로만 할 수 있다.
제일 처음에 점프를 할 때에는 한 칸밖에 점프하지 못한다. 즉, 1번 돌에서 2번 돌이 있는 곳으로 점프할 수 있다. 그 다음부터는 가속/감속 점프를 할 수 있는데, 이전에 x칸 점프를 했다면, 다음번에는 속도를 줄여 x-1칸 점프하거나, x칸 점프하거나, 속도를 붙여 x+1칸 점프를 할 수 있다. 물론 점프를 할 때에는 한 칸 이상씩 해야 한다.
각 돌들은 각기 그 크기가 다르고, 그 중 몇 개의 돌은 크기가 너무 작기 때문에 당신은 그러한 돌에는 올라갈 수 없다.
위와 같은 조건들을 만족하면서 1번 돌에서 N번 돌까지 점프를 해 갈 때, 필요한 최소의 점프 횟수를 구하는 프로그램을 작성하시오.

입력

첫째 줄에 두 정수 N, M(0 ≤ M ≤ N-2)이 주어진다. M은 크기가 맞지 않는, 즉 크기가 작은 돌의 개수이다. 다음 M개의 줄에는 크기가 작은 돌들의 번호가 주어진다. 1번 돌과 N번 돌은 충분히 크기가 크다고 가정한다.

출력

첫째 줄에 필요한 최소의 점프 횟수를 출력한다. 만약 N번 돌까지 점프해갈 수 없는 경우에는 -1을 출력한다.

import sys
input = sys.stdin.readline
sys.setrecursionlimit(10**4)

N, M = map(int, input().split())
MAX = 10001 # N(2 ≤ N ≤ 10,000)개 이므로
dp = [[-1]*150 for i in range(MAX)] 
# dp 초기값 -1로 설정
# 또한, N(2 ≤ N ≤ 10,000)개 기준으로 계산해서 150개 메모리 할당

ch = [0 for i in range(N+1)] ## 작은 돌 표시 위한 배열
flag = False 

for i in range(M): # 작은 돌 입력 
    idx = int(input())
    ch[idx] = 1 

def go(idx, jump): #dfs 탐색 / 중복 방문 방지를 위해 dp에 저장해주기
    global flag
    if idx == N: # idx는 모든 돌 위치/ 현재 돌 위치가 최종 돌 위치에 도달했다면
        flag = True # flag는 True로 바꿔주기
        return 0

    if dp[idx][jump] != -1:
        return dp[idx][jump]

    dp[idx][jump] = sys.maxsize
    for i in range(-1, 2): # -1, 0, 1  # x-1칸 또는 x칸 또는 x+1칸 이동하므로
        if jump+i >= 1: # 1이상이어야 앞으로 이동할 수 있음
            next = idx+(jump+i) # next는 현재 위치에서 이동한 만큼(jump+i) 더해준 위치
            if next <= N and ch[next] != 1: # 범위 내 있고, 작은 돌이 아니라면
                dp[idx][jump] = min(dp[idx][jump], 1+go(next, jump+i))

    return dp[idx][jump]

res = go(1, 0)

if flag: # flag값이 True
    print(res) 
else: # flag값이 False면 도달하지 못한 것이므로
    print(-1)

0개의 댓글