동적 계획법 예제 - 1
문제 분석
사용할 수 있는 3가지 연산(
/3,/2,-1)을바텀-업방식으로 구현할 수 있는지를 연습해볼 수 있는 문제다.주어진 조건을 점화식으로 변형해 코드화 해본다.
손으로 풀어보기
점화식의 형태와 의미를 도출한다.
dp[i]=i에서 1로 만드는 데 걸리는 최소 연산 횟수문제 요구사항을 그대로 dp 테이블에 적용
점화식을 구한다.
dp[i]=dp[i - 1] + 1=> 1을 빼는 연산if(i % 2 == 0) dp[i]=min(dp[i], dp[i / 2] + 1)=> 2로 나누는 연산if(i % 3 == 0) dp[i]=min(dp[i], dp[i / 3] + 1)=> 3으로 나누는 연산+ 1의 의미는 각i에서 1로 만들 수 있는 최소 연산 횟수에 현재 연산 한 번을 더한 것을 의미한다.
점화식을 이용해
dp테이블을 채운다.

dp[N]을 출력한다.
슈도코드
N(구하려고 하는 수)
dp(연산 횟수 최솟값을 저장하는 dp 테이블)
dp[1] = 0 # 1이 1이 될 때까지 연산이 불필요하므로 0으로 초기화 가능
for i 2~N:
dp[i] = dp[i - 1] + 1
if 2의 배수:
dp[i / 2] + 1이 dp[i] 보다 작으면 변경
if 3의 배수:
dp[i / 3] + 1이 dp[i] 보다 작으면 변경
dp[N] 출력코드 구현 - 파이썬
import sys
input = sys.stdin.readline
N = int(input())
dp = [0] * (N + 1)
dp[1] = 0 # 굳이 초기화할 필요는 없지만 명시적으로 표시
for i in range(2, N + 1):
dp[i] = dp[i - 1] + 1
if i % 2 == 0:
dp[i] = min(dp[i], dp[i // 2] + 1)
if i % 3 == 0:
dp[i] = min(dp[i], dp[i // 3] + 1)
print(dp[N])코드 구현 - 자바
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
public class Main {
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
int N = Integer.parseInt(br.readLine());
int[] dp = new int[N + 1];
dp[1] = 0;
for (int i = 2; i <= N; i++) {
dp[i] = dp[i - 1] + 1;
if (i % 2 == 0) {
dp[i] = Math.min(dp[i], dp[i / 2] + 1);
}
if (i % 3 == 0) {
dp[i] = Math.min(dp[i], dp[i / 3] + 1);
}
}
System.out.println(dp[N]);
}
}Last updated