#1463. 1로 만들기
0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 |
---|---|---|---|---|---|---|---|---|---|---|
0 | 0 | 1 | 1 | 2 | 3 | 2 | 3 | 3 | 2 | 3 |
0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 |
---|---|---|---|---|---|---|---|---|---|---|
0 | 0 | 1 | 1 | 2 | 3 | 2 | 3 | 3 | 2 | 3 |
📖문제
정수 X에 대한 연산
- X가 3으로 나누어 떨어지면, 3으로 나누다.
- X가 2로 나누어 떨어지면, 2로 나눈다.
- 1을 뺀다.
→ 정수 N이 주어졌을 때, 위의 연산 세 개를 적절히 사용해 1을 만든다. 연산을 사용하는 횟수의 최솟값을 출력한다.
입력1
1
2
출력1
1
10
입력2
1
1
출력2
1
3
🔍분석
👀 문제 유형 파악하기
1. 나열
먼저 나열해본다.
ex) n = 10
| 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | | — | — | — | — | — | — | — | — | — | — | — | | 0 | 0 | 1 | 1 | 2 | 3 | 2 | 3 | 3 | 2 | 3 |
- 2 : 2/2 (1)
- 3 : 3/3 (1)
- 4 : 4/2, 2/2 (2)
- 5: 5-1, 4/2, 2/2 (3) : -1하고 직전 수 더함
- 6 : 6/3, 3/3 (2) or 6/2, 3/3 (2)
- 7 : 7-1, 6/3, 3/3 (3) : -1하고 직전 수 더함
- 8 : 8/2, 4/2, 2/2 (3)
- 9 : 9/3, 3/3 (2)
- 10 : 10-1, 9/3, 3/3(3) or 10/2, 5-1, 4/2, 2/2 (4)
- 11 : 11-1, 10-1, 9/3, 3/3 (4) : -1하고 직전 수 더함
- 12 : 12/3, 4/2, 2/2 (3) , 12/2, 6/2, 3/3(3), 12/2, 6/3, 2/2 (3)
- 13 : 13-1, 12/3, 4/2, 2/2 (4) : -1하고 직전 거 더함
- 14: 14/2, 7-1, 6/3, 2/2 () ⇒ 횟수 4
- 15 : 15-1, 14/2, 7-1, 6/3, 2/2 (5) or 15/3, 5-1, 4/2, 2/2 (4)
- 16 : 16/2, 8/2, 4/2, 2/2 (4)
- 17 : 17-1, 16/2, 8/2, 4/2, 2/2 (5) : -1하고 직전 수 더함
2. 규칙성 발견
열거함으로써 규칙성을 발견했다.
- 2의 배수 → 2로 나누어 떨어질 때까지의 연산횟수 ⇒ i % 2==0
- 3의 배수 → 3으로 나누어 떨어질 때까지의 연산횟수
- 2와 3의 공배수 → 2나 3으로 나누어 떨어질 때까지의 연산횟수
- 2와 3으로 나누어 떨어지지 않는 수 → -1 + 직전 횟수
- 하면 나누어 떨어지는 수로 만들어진다. 바로 그 직전 수가 되기 때문에 그 직전 수의 연산횟수를 더한다.
- 번외 ) 2와 3 모두 나누기가 가능할 때, 3으로 먼저 나누는 것이 연산횟수가 더 적어진다. if문 사용할 때 3으로 나누는 연산을 먼저 한다.
하지만 예외 존재!
위에서 나열했을 때, 10의 값은 2로 나눈 값이 아니라 먼저 -1을 빼야 최소횟수가 나온다.
- 10 : 10-1, 9/3, 3/3(3) or 10/2, 5-1, 4/2, 2/2 (4)
하지만 15의 경우는 먼저 1을 뺀 것이 아니라 3으로 나누고 나서 연산하는 것이 최소횟수가 나온다.
- 15 : 15-1, 14/2, 7-1, 6/3, 2/2 (5) or 15/3, 5-1, 4/2, 2/2 (4)
⇒ 따라서, 무조건적으로 -1을 먼저 연산하거나, 나누기 연산을 먼저 하는 것이 아니라, 두 연산 모두 거친 후 둘 중의 최솟값을 비교하는 방식으로 가야 한다. (min함수 사용)
👀 리스트값과 인덱스 변수의 의미 파악하기
→ 각각의 테이블에는 해당 인덱스의 숫자가 1이 된다. 이 1이 되는데 필요한 연산의 수가 저장된다.
- 테이블에 들어가야 할 것 : 최소 턴 개수 (출력하기 위함)
- 테이블 인덱스가 의미하는 것 : 1~n
index (1~n) | 1 | 2 | 3 | … | n |
---|---|---|---|---|---|
최소 턴 개수 |
💡풀이
Bottom Up 방식
1
2
3
4
5
6
7
8
9
10
11
n = int(input()) # 수 입력 받기
dp = [0] * (n+1) #dp 리스트 초기화
for i in range(2, n+1): #2부터 n까지 반복
dp[i] = dp[i-1] + 1 #1을 빼는 연산 (1회 진행)
if i % 3 == 0: # 3으로 나누어 떨어질 때, 3으로 나누는 연산
dp[i] = min(dp[i], dp[i//3]+1)
if i % 2 == 0: #2로 나누어 떨어질 때, 2로 나누는 연산
dp[i] = min(dp[i], dp[i//2]+1)
print(dp[n]) #출력
- 변수
n
에 입력 받는다 dp리스트
에 0이n+1
개 있는 리스트로 초기화한다- dp는 인덱스의 숫자가 1이 되는데 필요한 연산의 최솟값이 들어간다.
- 즉, dp[1] = 0이다. 1이 1로 되는데 필요한 연산횟수는 0이기 때문이다.
- dp[2] = 1이다. 2가 1로 되기 위한 연산은 2/2인 1이 된다.
- dp는 인덱스의 숫자가 1이 되는데 필요한 연산의 최솟값이 들어간다.
for i in range (2, n+1)
: 2부터 n까지 i로 반복한다.- 이때 2부터 시작하는 이유는 0과 1은 연산하지 않아도 0이기 때문이다.
dp[i] = dp[i-1] + 1 : dp[1]
는 숫자가 i가 1이 되는데 걸리는 최소한의 연산횟수를 저장해야 한다.- i에서 1을 빼면 i-1이 된다.
- dp[i-1] 에 +1을 하여,
dp[i-1]
(i=1이 1이 되는데 필요한 최소한의 연산)+ 1
(i에서 1을 빼서 i-1이 되는데 필요한 연산 횟수 1회)로 초기화한다.
**if i % 3 == 0: dp[i] = min(dp[i], dp[i//3]+1)**
:: i가 3으로 나누어 떨어질 때,dp[i]
와dp[i//3] + 1
중 최솟값을 dp[i]에 저장한다.- dp[i]는 위에서 초기화한 dp[i-1]의 값이 들어가 있다.
dp[i//3]+1
은 i를 3으로 나눈 값이 1이 되는데 걸린 최소 연산횟수 + i를 3으로 나눈 연산횟수 1회이다.
- dp[i]는 위에서 초기화한 dp[i-1]의 값이 들어가 있다.
if i%2==0: d[i]=min(d[i],d[i//2]+1)
i가 2로 나누어 떨어질 때, dp[i]와 dp[i//2]+1 중 최솟값을 dp[i]에 저장한다. 위의 연산과 과정이 동일하다.- 이렇게 for문을 반복하면 dp[n]에는 n이 1이 되는데 필요한 연산의 최소 횟수가 들어간다. 따라서 dp[n]을 출력하면 된다
다른 풀이 참고
[백준] 1463 1로 만들기 (DP) - Python / 자세한 설명 / 여러가지 풀이 / 실버1
🤔회고
- DP 문제를 드디어 혼자 힘으로 80%정도 풀었다. 이전까지는 헷갈렸는데, 이전보다는 문제 푸는 힘이 늘었다.
- 경우의 수가 많은 문제는 일단 나열해본다! 나열하고나서 규칙성이 보이거나 어떠한 수들이 반복이 된다면 DP를 고려해보자