알고리즘
BOJ)괄호 추가하기 3
1일1공부실천하자
2024. 6. 28. 23:50
https://www.acmicpc.net/problem/16639
시간복잡도를 확인하고 브루트포스로 해결하려했지만 실패했다.
결국 다른 사람의 코드를 참고하여 풀었고, 이해하는데 조금 애를 먹었다.
N = int(input())
E = input()
M = N // 2 + 1
max_dp = [[-10 ** 9] * M for _ in range(M)]
min_dp = [[10 ** 9] * M for _ in range(M)]
for i in range(M):
max_dp[i][i] = min_dp[i][i] = int(E[i * 2])
for k in range(1,M):
for i in range(M - k):
j = i + k
for x in range(i, j):
#최댓값
max_dp[i][j] = max(max_dp[i][j], eval(str(max_dp[i][x]) + E[x * 2 + 1] + str(max_dp[x + 1][j])))
max_dp[i][j] = max(max_dp[i][j], eval(str(min_dp[i][x]) + E[x * 2 + 1] + str(max_dp[x + 1][j])))
max_dp[i][j] = max(max_dp[i][j], eval(str(max_dp[i][x]) + E[x * 2 + 1] + str(min_dp[x + 1][j])))
max_dp[i][j] = max(max_dp[i][j], eval(str(min_dp[i][x]) + E[x * 2 + 1] + str(min_dp[x + 1][j])))
#최솟값
min_dp[i][j] = min(min_dp[i][j], eval(str(max_dp[i][x]) + E[x * 2 + 1] + str(max_dp[x + 1][j])))
min_dp[i][j] = min(min_dp[i][j], eval(str(min_dp[i][x]) + E[x * 2 + 1] + str(max_dp[x + 1][j])))
min_dp[i][j] = min(min_dp[i][j], eval(str(max_dp[i][x]) + E[x * 2 + 1] + str(min_dp[x + 1][j])))
min_dp[i][j] = min(min_dp[i][j], eval(str(min_dp[i][x]) + E[x * 2 + 1] + str(min_dp[x + 1][j])))
print(max_dp[0][-1])
max_dp[i][j]는 i부터 j까지(숫자들)의 계산 중 가장 큰 값이 들어가고, min_dp[i][j]는 i부터 j까지의 수의 계산 중 가장 작은 값이 들어간
다.
이 문제의 핵심은 반례를 생각하는 것이다.
일반적으로 큰 숫자 * 큰숫자로 가장 큰 숫자를 만들 수 있지만,계산하여 나온 마이너스 값 * 계산하여 나온 마이너스 값의 경우 가장 큰 값이 될 수 있기 때문이다.
때문에 계산 값중 가장 큰 값과 가장 작은 값을 따로 저장해야 이 문제를 풀 수가있다.
우선 코드를 보면 이해가 안되지만 직접 반복문의 로그를 찍어보면 이해가 간다.
for i in range(M):
max_dp[i][i] = min_dp[i][i] = int(E[i * 2])
우선 위 코드로 dp배열에 n으로 받은 수를 넣어준다.
그럼 위와 같은 배열이 된다.
우선 max,min은 생각하지 않겠다.
그 다음 반복문 하나하나를 보며 i,j,x,min_dp[i][j], max_dp[i][j]의 값을 보면 다음과 같다.
위 코드를 보면 다음과 같은 규칙이 존재한다.
이때 중요한 것은 min_dp와 max_dp의 [i][j]는 각각 max_dp와 min_dp를 비교해가며 최소/최대값을 갱신한다.
이런 식으로 max_dp[0][-1]는 3부터 2까지의 계산 값 중 가장 큰 값이 들어가게된다.