이해하는데 꽤 오래 걸렸다.
dp를 대각선으로 그려야 답을 구할 수 있는 문제로 아주 신박하고 연습이 필요한 문제일 것 같다.
dp[i][i+j]로 놓고 i를 증가시키면 양쪽이 1,1씩 증가해 대각선 배열을 먼저 연산 할 수 있으며 j가 늘어나면 가운데 대각선에서 한칸 더 멀어진 배열을 연산하게 되는 식이다.
즉 2개씩 묶은 행렬을 계산하고(AB,BC,CD,CE)
그 값을 가지고 3개씩 묶은 행렬을 계산. min(AB*C or A*BC)
그걸 가지고 4개씩 묶은 행렬을 계산min(A*BCD,AB*CD,ABC*D)
N*N 인접 행렬에서 대각선으로 배열을 채우는법을 이해할 수 있었고 차후 다른 문제에서도 쓸만한아이디어인 것 같다.
행렬의 곱셈이 이루어지는 숫자는 [앞행렬의 행값 * 앞행렬의 열값(==뒷행렬의 행값) * 뒷행렬의 열값]로
ABCD*E라면 [A의 행값*D의 열값*E의 열값]을 기존 ABCD계산 최솟값에 더해주면 된다.
추후 대각배열이 나오면 다시 들여다 볼 것.
import sys
input = sys.stdin.readline
N = int(input())
matrix = [list(map(int, input().split())) for i in range(N)]
# dp[i][j]는 i번재 행렬에서 j번째 행렬까지 곱한
dp = [[0] * N for _ in range(N)]
for i in range(1, N): #몇 번째 대각선? 0번째 대각선은 자기자신이므로 0유지.
for j in range(0, N-i): #대각선에서 몇 번째 열?(N*N에서 반만 채우므로 대각선 번호가 끝으로 갈수록 계산해야되는 칸도 하나씩 줄어감)
x = i+j # 차이가0인 본인대각선 모든값 0 -> 첫 대각선 계산수 N-1 -> 두번째 대각선 계산수 N-2 .... N번째 대각선 1회.
if i == 1: # x는 i번째 대각선의 j번째 열. 대각선은 한칸마다 1,1씩 증가해야하므로 j가 양쪽에 들어가 1씩 증가시켜준다.i는 표준편차의 크기.
dp[j][x] = matrix[j][0] * matrix[j][1] * matrix[x][1] # 계산 횟수는 가운데 낀 숫자는 1번 맨앞, 맨뒤숫자 1번씩 곱.
continue
dp[j][x] = 2**32 #최댓값을 미리 넣어줌(min연산을 위해 inf값 할당해주듯)
#k는 A~E중 어디를 자르는가의 문제. ex ABC라면 AB,C / A,BC
#즉 k에따라 콤마위치가 변하고 dp[j][k]는 j부터 k행렬까지의 최소값, dp[k+1][x]는 K+1에서 x까지의 최솟값
#뒤에 곱하는 dp는 앞행렬과 뒷행렬을 곱하는 횟수(A의 행수*자르는index인 k의 열수*마지막인자인 x의 열수)
for k in range(j, x):
dp[j][x] = min(dp[j][x], dp[j][k] + dp[k+1][x] + matrix[j][0] * matrix[k][1] * matrix[x][1])
print(dp[0][N-1]) #맨 오른쪽 위
'practivceAlgorithm > 백준' 카테고리의 다른 글
[백준][Python] 12865 평범한 배낭 (0) | 2021.07.25 |
---|---|
[백준][Python] 11660 구간 합 구하기 (0) | 2021.07.25 |
[백준][Python] 14888 연산자 끼워넣기 (0) | 2021.07.25 |
[백준][Python] 21278 호석이 두 마리 치킨 (0) | 2021.07.25 |
[백준][Python] 2633 음악프로그램 (1) | 2021.07.25 |