알고리즘

BOJ) 동전 1. Python

1일1공부실천하자 2024. 6. 24. 21:53

 

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

 

 

 

 

입력 예제로 들어온 값으로 예를 들어보자

3 10
1
2
5

 

1,2,5의 동전으로 10을 만들 수 있는 경우의 수는 몇 개인가?

우리가 생각해보아야할 것은 

  1. 1의 동전으로 몇 개를 만들 수 있는가.
  2. 1과 2의 동전으로 몇 개를 만들 수 있는가
  3. 1과 2와 5의 동전으로 몇 개를 만들 수 있는가

여느 dp문제와 헷갈린 점이

2원짜리 동전은 1원짜리 동전으로 나눌 수 있으니 이를 이용해 dp를 작성하면 된다 생각했다.

하지만 이는 문제를 잘못 쪼갠 것이다.

 

 

1원으로 k원을 만들 수 있는 경우의 수,

1과 2원으로 k원을 만들 수 있는 경우의 수, 

1과 2와 5원으로 k원을 만들 수 있는 경우의 수

이런 작은 문제들로 쪼개야한다.

 

우선 1원으로 0원부터 k원(10원)을 만들 수 있는 경우의 수는 다음과 같다.

 

 

0원은 동전을 아예 제출하지 않으면 0원을 만들 수 있으니 1가지의 경우의 수가 존재한다.

 

1원과 2원은 다음과 같다.

 

 

2원의 경우 현재 x원을 만들 수 있는 경우의 수는 다음과 같다.

  1. 2원으로 x원을 만들 수 있는가?
    1. 맞다면 경우의 수 + 1
  2. 1원으로 x원을 만들 경우의 수 ( +1)
  3. 2원과 1원으로 x원을 만들 경우의 수

 

즉, 위 사진 같이 위 1원으로 만든 경우의 수를 상속받는다.

여기서 2원과 1원으로 x원을 만들 수 있는 경우의 수를 구하는 과정은 문제를 더 쪼개보면 가능하다.

x를 4라고 가정한다면,

  1. 1원으로 4원을 만드는 경우의 수를 상속받으며
  2. 1원과 2원으로 2를 만드는 경우의 수를 더한다.

이를 5원까지 적용해보면 다음과 같다.

 

 

위에는 어떤 공식이 있는데,

x원이라 가정했을 때, dp의 x - coin 인덱스 번째를 더한다.

이를 점화식으로 표현하면 다음과 같다.

 

for num in arr:
    for idx in range(num, k + 1):
        dp[idx] = dp[idx] + dp[idx - num]

 

n, k = map(int, input().split())

arr = []

dp = [0] * (k+1)
dp[0] = 1

for _ in range(n):
    arr.append(int(input()))

for num in arr:
    for idx in range(num, k + 1):
        dp[idx] = dp[idx] + dp[idx - num]


print(dp[k])