알고리즘/백준

[백준 12865] 평범한 배낭 (C++)

겜도리도리 2021. 8. 29. 21:30
반응형

문제

https://www.acmicpc.net/problem/12865

 

12865번: 평범한 배낭

첫 줄에 물품의 수 N(1 ≤ N ≤ 100)과 준서가 버틸 수 있는 무게 K(1 ≤ K ≤ 100,000)가 주어진다. 두 번째 줄부터 N개의 줄에 거쳐 각 물건의 무게 W(1 ≤ W ≤ 100,000)와 해당 물건의 가치 V(0 ≤ V ≤ 1,000)

www.acmicpc.net

풀이

유명한 0-1 배낭 문제입니다.

dp[i][j]의 의미는 i개까지의 물건을 넣는 경우에서 무게가 j일때 가치의 최대값입니다.

예제 입력으로 분석해보겠습니다.

n / k 1 2 3 4 5 6 7
0 0 0 0 0 0 0 0
1 0 0 0 0 0 13 13
2 0 0 0 8 8 13 13
3 0 0 6 8 8 13 14
4 0 0 6 8 12 13 14

먼저 n = 0일때는 넣을 수 있는 물건이 없으므로 모든 무게에서 가치는 0입니다.

n = 1 일 때는 무게가 6, 가치가 13인 물건만 넣을 수 있으므로 1~5무게에서는 가치 0, 6~7무게에서는 가치 13입니다.

n = 2 일 때는 무게가 4, 가치가 8인 물건이 추가됩니다. 아래의 점화식에 따라 dp[i][j]의 값을 결정합니다.

1) 물건을 넣을 수 없는 경우에는 dp[i-1][j]의 값을 그대로 넣습니다.

2) 물건을 넣을 수 있는 경우에는 dp[i-1][j]의 값과 dp[i-w[i]]+v[i]의 값을 비교하여 더 큰 값을 넣습니다.

이 과정을 반복했을 때 dp[n][k]의 값이 가치의 최댓값이 됩니다.

소스 코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
#include<iostream>
#include<algorithm>
#include<vector>
using namespace std;
 
int dp[101][100001];
int w[101];
int v[101];
int n, k;
 
int main()
{
    cin >> n >> k;
    for (int i = 1; i <= n; i++)
    {
        cin >> w[i] >> v[i];
    }
    for (int i = 1; i <= n; i++)
    {
        for (int j = 1; j <= k; j++)
        {
            //물건을 넣을 수 있는 경우
            if (j >= w[i])
                dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - w[i]] + v[i]);
            //물건을 넣을 수 없는 경우
            else
                dp[i][j] = dp[i - 1][j];
        }
    }
    cout << dp[n][k];
    return 0;
}
cs
반응형

'알고리즘 > 백준' 카테고리의 다른 글

[백준 20364] 부동산 다툼 C++  (0) 2021.09.01
[백준 3079] 입국심사 (C++)  (0) 2021.08.30
[백준 5430] AC (C++)  (0) 2021.08.29
[백준 11000] 강의실 배정 (C++)  (0) 2021.08.27
[백준 11559] Puyo Puyo (C++)  (0) 2021.08.27