일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
- 네이버 부스트캠프
- ObservableObject
- ObservedObject
- Apple Developer Academy @ POSTECH
- 애플 아카데미 후기
- Swift 디자인패턴
- 애플 디벨로퍼 아카데미 후기
- 치지직
- 소프트웨어분석및설계
- swift문법
- Swift 기능
- 네이버 치지직
- global soop
- 데이터베이스
- useReducer
- react
- 앱 비교 프로젝트
- iOS 개발 오류
- SWIFT
- OS
- apple developer academy 후기
- sqoop
- 데이터베이스 공부
- StateObject
- 운영체제
- 제앱소
- Swift 문법
- 숭실대
- 애플 디벨로퍼 아카데미 21주차 회고
- 애플 디벨로퍼 아카데미
- Today
- Total
사과하는 제라스
[백준 BOJ 12865번] 평범한 배낭 본문
목차
출처 : https://www.acmicpc.net/problem/12865
1. 문제
이 문제는 아주 평범한 배낭에 관한 문제이다.
한 달 후면 국가의 부름을 받게 되는 준서는 여행을 가려고 한다. 세상과의 단절을 슬퍼하며 최대한 즐기기 위한 여행이기 때문에, 가지고 다닐 배낭 또한 최대한 가치 있게 싸려고 한다.
준서가 여행에 필요하다고 생각하는 N개의 물건이 있다. 각 물건은 무게 W와 가치 V를 가지는데, 해당 물건을 배낭에 넣어서 가면 준서가 V만큼 즐길 수 있다. 아직 행군을 해본 적이 없는 준서는 최대 K만큼의 무게만을 넣을 수 있는 배낭만 들고 다닐 수 있다. 준서가 최대한 즐거운 여행을 하기 위해 배낭에 넣을 수 있는 물건들의 가치의 최댓값을 알려주자.
2. 입력
첫 줄에 물품의 수 N(1 ≤ N ≤ 100)과 준서가 버틸 수 있는 무게 K(1 ≤ K ≤ 100,000)가 주어진다. 두 번째 줄부터 N개의 줄에 거쳐 각 물건의 무게 W(1 ≤ W ≤ 100,000)와 해당 물건의 가치 V(0 ≤ V ≤ 1,000)가 주어진다.
입력으로 주어지는 모든 수는 정수이다.
4 7
6 13
4 8
3 6
5 12
3. 출력
한 줄에 배낭에 넣을 수 있는 물건들의 가치합의 최댓값을 출력한다.
14
4. 풀이
일단 0/1 knapsack 문제이기 때문에 한 물건을 쪼개지 못하고 그대로 담아야 한다. 우선 dp[i][j]에서 j 무게만큼 채울 수 있을 때, 1~i번째 물건들을 가지고서만 만든 최적의 해이다.
<Top-Down 방식>
만약 i가 0 이하면 채울거리가 없기에 0을 리턴하고, k=0일 경우에는 알아서 dp[0][0]까지 내려오기 때문에 dp[0][0]만 0으로 설정해두면 된다. 만약 i가 0보다 크다면 그때부턴 dp값이 있으면 그대로 출력하고, 아니라면 i번째 물건보다 가방용량이 작다면 i-1번째에 담을 수 있었던 경우를 dp에 쓰고, 가방용량이 더 크다면 i번째 이전까지 현 무게에 넣을 수 있는 최적의 해(knapsack[i-1][j])와 지금 물건을 담고나서 이전의 물건들에 대해 최적의 해를 현 물건의 가치와 더한 값(knapsack[i-1][j-w[i]]+v[i])을 비교한 후 큰 값을 dp에 쓴다.
<Bottom-Up 방식>
Top-Down방식과 똑같으나 이중 for문을 이용해서 dp값들을 채워 나간다.
5. 소스코드
<Top-Down 방식>
import java.io.*;
import java.util.*;
public class Main {
static int N;
static int K;
static int [] w;
static int [] v;
static Integer [][] dp;
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
StringTokenizer st;
st = new StringTokenizer(br.readLine(), " ");
N = Integer.parseInt(st.nextToken());
K = Integer.parseInt(st.nextToken());
w = new int [N+1];
v = new int [N+1];
for(int i=1; i<=N; i++){
st = new StringTokenizer(br.readLine(), " ");
w[i] = Integer.parseInt(st.nextToken());
v[i] = Integer.parseInt(st.nextToken());
}
dp = new Integer [N+1][K+1];
System.out.println(knapsack(N, K));
}
public static int knapsack(int i, int k){
if(i <= 0){
return 0;
}
else{
if(dp[i][k] == null){
if(w[i] > k)
dp[i][k] = knapsack(i-1, k);
else{
dp[i][k] = (int)Math.max(knapsack(i-1, k), knapsack(i-1, k-w[i])+v[i]);
}
}
}
return dp[i][k];
}
}
<Bottom-Down 방식>
import java.io.*;
import java.util.*;
public class Main {
static int N;
static int K;
static int [] w;
static int [] v;
static int [][] dp;
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
StringTokenizer st;
st = new StringTokenizer(br.readLine(), " ");
N = Integer.parseInt(st.nextToken());
K = Integer.parseInt(st.nextToken());
w = new int [N+1];
v = new int [N+1];
for(int i=1; i<=N; i++){
st = new StringTokenizer(br.readLine(), " ");
w[i] = Integer.parseInt(st.nextToken());
v[i] = Integer.parseInt(st.nextToken());
}
dp = new int [N+1][K+1];
for(int i=1; i<=N; i++){
for(int j=1; j<=K; j++){
dp[i][j] = 0;
if(w[i] > j) dp[i][j] = dp[i-1][j];
else{
dp[i][j] = (int)Math.max(dp[i-1][j], dp[i-1][j-w[i]]+v[i]);
}
}
}
System.out.println(dp[N][K]);
}
}
6. 배운 것
일단, 처음에 문제를 잘못 이해했었다. 0/1 knapsack 문제로만 이해를 하고 주어진 물건들이 (1종류-1개씩) 이란걸 인지하지 못한 채 풀이한지라 다음 부분에서 i-1을 이해하지 못했었다.
else{
dp[i][k] = (int)Math.max(knapsack(i-1, k), knapsack(i-1, k-w[i])+v[i]);
knapsack 문제의 조건에서 만약 종류당 1개씩만이란 조건이 없다면 i-1만 i로 바꿔주어서 중복사용하는 방식의 결과도 도출할 수 있다.
이 문제에서의 Top-Down 방식이 사실상 DP문제를 푸는데에 있어서 찐 Top-Down 방식이라고 볼 수 있다.
'JAVA 백준 알고리즘 문제풀이 > 동적 계획법' 카테고리의 다른 글
[백준 BOJ 9251번] LCS (0) | 2021.12.11 |
---|---|
[백준 BOJ 2565번] 전깃줄 (0) | 2021.12.10 |
[백준 BOJ 11054번] 가장 긴 바이토닉 부분 수열 (0) | 2021.12.05 |
[백준 BOJ 11053번] 가장 긴 증가하는 부분 수열 (0) | 2021.12.05 |
[백준 BOJ 10844번] 쉬운 계단 수 (0) | 2021.12.05 |