Algorithm/Online judge

[백준] 11048번 > 이동하기

민철킹 2021. 3. 16. 17:30

www.acmicpc.net/problem/11048

 

11048번: 이동하기

준규는 N×M 크기의 미로에 갇혀있다. 미로는 1×1크기의 방으로 나누어져 있고, 각 방에는 사탕이 놓여져 있다. 미로의 가장 왼쪽 윗 방은 (1, 1)이고, 가장 오른쪽 아랫 방은 (N, M)이다. 준규는

www.acmicpc.net

문제

준규는 N×M 크기의 미로에 갇혀있다. 미로는 1×1크기의 방으로 나누어져 있고, 각 방에는 사탕이 놓여져 있다. 미로의 가장 왼쪽 윗 방은 (1, 1)이고, 가장 오른쪽 아랫 방은 (N, M)이다.

준규는 현재 (1, 1)에 있고, (N, M)으로 이동하려고 한다. 준규가 (r, c)에 있으면, (r+1, c), (r, c+1), (r+1, c+1)로 이동할 수 있고, 각 방을 방문할 때마다 방에 놓여져있는 사탕을 모두 가져갈 수 있다. 또, 미로 밖으로 나갈 수는 없다.

준규가 (N, M)으로 이동할 때, 가져올 수 있는 사탕 개수의 최댓값을 구하시오.

입력

첫째 줄에 미로의 크기 N, M이 주어진다. (1 ≤ N, M ≤ 1,000)

둘째 줄부터 N개 줄에는 총 M개의 숫자가 주어지며, r번째 줄의 c번째 수는 (r, c)에 놓여져 있는 사탕의 개수이다. 사탕의 개수는 0보다 크거나 같고, 100보다 작거나 같다.

출력

첫째 줄에 준규가 (N, M)으로 이동할 때, 가져올 수 있는 사탕 개수를 출력한다.

예제 입력 1

3 4

1 2 3 4

0 0 0 5

9 8 7 6

예제 출력 1

31

예제 입력 2

3 3

1 0 0

0 1 0

0 0 1

예제 출력 2

3

예제 입력 3

4 3

1 2 3 6

5 4 7 8

9 12 11 10

예제 출력 3

47

 


풀이

 

1 0 0
0 1 0
0 0 1

위와 같은 미로가 있다고 가정하겠다. 현재 위치에서 이동할 수 있는 경우는

오른쪽, 오른쪽 대각선 아래, 아래쪽 3가지 경우이다.

즉, 아래에 빨간 펜으로 표시한 위치에서의 최댓값은 파란색 위치 3곳 중 제일 큰 값과 현재 나의 값을 더한 것이다.

이를 점화식으로 표현한다면?

dp[i][j] = max(dp[i-1][j], dp[i][j-1], dp[i-1][j-1]) + maze[i][j-1]

위와 같이 표현할 수 있을 것이다.

 

하지만 예외 case가 있다. 1행의 경우에는 나머지 2곳이 좌표계 밖이고 오로지 오른쪽으로만 이동할 수 있다.

1열의 경우에는 아래쪽으로만 이동할 수 있다. 따라서 1행과 1열을 미리 설정해주고 dp를 진행한다.

1 1 1
1 0 0
1 0 0

위와 같이 dp리스트를 수정해준다음 점화식을 진행

n, m = map(int, input().split())
maze = [list(map(int, input().split())) for _ in range(n)]
dp = [[0] * (m+1) for _ in range(n)]
for i in range(1, m+1):
    dp[0][i] += (maze[0][i-1] + dp[0][i-1]) # 1행
for i in range(1, n):
    dp[i][1] += (maze[i][0] + dp[i-1][1]) # 1열

for i in range(1, n):
    for j in range(2, m+1):
        dp[i][j] = max(dp[i-1][j], dp[i][j-1], dp[i-1][j-1]) + maze[i][j-1]
print(dp[-1][-1])

 

반응형

'Algorithm > Online judge' 카테고리의 다른 글

[백준] 1932번 > 정수 삼각형  (0) 2021.03.22
[백준] 10451번 > 순열 사이클  (0) 2021.03.19
[백준] 16916번 > 부분 문자열  (0) 2021.03.14
[백준] 14490번 > 백대열  (0) 2021.03.14
[백준] 7576번 > 토마토  (0) 2021.03.14