문제
어떤 투자가가 여러 기업들에게 돈을 투자해서 최대의 이익을 얻고자 한다. 단, 투자는 만원 단위로 할 수 있으며 각 기업은 많이 투자할수록 많은 이익을 투자가에게 돌려준다. 돈을 투자하지 않은 경우는 당연히 얻게 되는 이익도 없다. 예를 들어서, 한 투자가가 4만원을 갖고 두 개의 기업들에 각각 만원 단위로 투자했을 경우 얻을 수 있는 이익은 다음과 같다.
투자 액수 (만원) | 기업 A | 기업 B |
---|---|---|
1 | 5 | 1 |
2 | 6 | 5 |
3 | 7 | 9 |
4 | 8 | 15 |
위의 경우 만일, 기업 A에 1만원, 기업 B에 3만원을 투자하는 경우 투자가가 얻는 이익은 14만원(5만원+9만원)이다. 4만원을 투자해서 가장 많은 이익을 얻을 경우 기업 B에만 4만원을 투자하는 경우로서 이때의 이익은 15만원이다. 여기서 투자가는 한 기업에 돈을 나누어 투자할 수는 없다.
투자액이 정해져 있고, 기업의 개수와 각 기업에 투자했을 경우에 얻게 되는 이익이 주어졌을 때 가장 많은 이익을 얻을 수 있는 투자방식과 이때의 이익금을 구하는 프로그램을 작성하라.
입력
첫째 줄에 투자 금액 N과 투자 가능한 기업들의 개수 M이 주어진다. (1 ≤ N ≤ 300, 1 ≤ M ≤ 20)
둘째 줄부터 N개의 줄에 투자액수와 각 기업이 투자가에게 주는 이익이 주어진다. 투자 금액은 항상 1보다 크거나 같고, N보다 작거나 같고, 같은 투자 금액이 두 번 이상 주어지는 경우는 없다. 즉, i번 줄에 주어지는 투자 금액은 i-1만원이다.
4 2
1 5 1
2 6 5
3 7 9
4 10 15
출력
첫 줄에 얻을 수 있는 최대 이익을 출력하고, 둘째 줄에는 각 기업에 투자한 액수를 출력한다. 최대 이익은 231보다 작다.
15
0 4
풀이
from sys import stdin
input = stdin.readline
N, M = map(int, input().split())
benefit_info = [[0] * (M+1)] + [list(map(int, input().split())) for _ in range(N)]
DP = [[0] * (M + 1) for _ in range(N + 1)]
investments = [[0] * (M + 1) for _ in range(N + 1)]
def cal_maximum_benefit(price, business):
MAX = benefit_info[price][business]
cur_investment = price
for i in range(price):
temp = DP[price - i][business - 1] + benefit_info[i][business]
if MAX < temp:
MAX = temp
cur_investment = i
return MAX, cur_investment
for i in range(1, N + 1):
for j in range(1, M + 1):
DP[i][j], investments[i][j] = cal_maximum_benefit(i, j)
print(DP[N][M])
result = [0] * M
for business in range(M-1, -1, -1):
result[business] = investments[N][business+1]
N -= investments[N][business+1]
print(*result)
코드 해설
다음은 문제 풀이의 핵심 코드이다.
def cal_maximum_benefit(price, business):
MAX = benefit_info[price][business]
cur_investment = price
for i in range(price):
temp = DP[price - i][business - 1] + benefit_info[i][business]
if MAX < temp:
MAX = temp
cur_investment = i
return MAX, cur_investment
모든 경우의 수를 돌면서, 돈(price)을 1번부터 business번 기업들 중 하나 이상의 기업에 투자하였을 때, 얻을 수 있는 최대 이익(MAX)과 그때 business번 기업에 투자해야할 금액(cur_investment)을 구한다.
코드를 자세히 뜯어보자
MAX = benefit_info[price][business]
cur_investment = price
먼저 최댓값을 모든 돈(price)을 현재 기업에 투자했을 때 얻는 이익으로 초기화한다.
for i in range(price):
temp = DP[price - i][business - 1] + benefit_info[i][business]
if MAX < temp:
MAX = temp
cur_investment = i
그리고 현재기업에 0원부터 price - 1원까지 투자하고,
남은 돈(price - i원)을 지금까지 기업의 앞의 기업들에게 투자를 하였을 때의 이득을 temp에 담는다.
만약 temp가 MAX보다 크다면 MAX를 갱신해주고 이때의 기업(business)에 투자한 금액을 cur_invest에 저장한다.
이를 1원부터 price(전역변수 i)원까지, 1번기업부터 business(전역변수 j)번 기업까지 반복해준다.
for i in range(1, N + 1):
for j in range(1, M + 1):
DP[i][j], investments[i][j] = cal_maximum_benefit(i, j)
DP[i][j]는 i원을 1번 ~ j번 기업들에게 투자하였을 때, 얻을 수 있는 최대 금액이 된다.
investments[i][j]는 i원을 1번 ~ j번기업들에게 투자하여 최대 금액을 얻었을 때 j번 기업에 투자해야될 금액이 된다.
다음은 역추적이다.
result = [0] * M
for business in range(M, 0, -1):
result[business - 1] = investments[N][business]
N -= investments[N][business]
모든 기업에 0원을 투자한 것으로 초기화를 한 뒤,
최대 투자 금액인 N부터 시작하여 N원을 business에 투자한것이므로 투자한 금액을 N에서 빼준다.
그러면 business번 기업을 제외한 business - 1번 기업들 까지 중에 남은 N원에서 투자할 금액인 investments[N][business]원을 다시 빼주는 것을 모든 기업에 반복한다.
ps. 풀이의 코드와 for문이 다른데, 설명을 작성하다가 현재처럼 바꾸는 것이 이해가 더 쉬울 것 같아 바꾸었다.
+1연산 두 번에서 -1 연산 한 번으로 바꿨는데 이상하게 시간이 느려졌다...?
M이 최대 20인데 시간 차이가 왜 나지....
일단 풀이는 더 빠른 코드로 작성하여두었다.
왜냐하면 백준 1등하고 신났기 때문에
'알고리즘과 자료구조' 카테고리의 다른 글
트라이(Trie): 문자열 탐색과 저장을 효율적으로 만들어주는 자료구조 (0) | 2024.12.08 |
---|---|
[프로그래머스] 경주로 건설(2020 카카오 인턴십 기출 문제) JS 풀이(부제: 자바스크립트로 최소힙 자료 구조와 다익스트라 알고리즘 구현하기) (0) | 2024.10.06 |
[BOJ] 백준 1305번 광고 Python (0) | 2023.11.27 |