2021. 2. 27. 17:41ㆍ코딩 테스트/실전 문제
1. 문제
이상한 계산기는 두 가지 버튼과 숫자를 출력하는 화면으로 구성되어 있다. 숫자를 출력하는 화면은 총 5자리의 정수를 표현할 수 있다. 각 버튼은 현재까지의 계산 결과에 어떠한 연산을 할지를 나타내는 것으로, 각 버튼은 다음과 같이 연산을 수행한다.
- Mul : 현재까지의 계산 결과에 2를 곱한다
- Div : 현재까지의 계산 결과를 3으로 나눈다. 단, 결과가 정수가 아닐 경우, 소수점 이하를 모두 버린다. 계산기를 작동시키면, 현재까지의 계산 결과로써 1이 출력된다. 이후 사용자가 어떻게 버튼을 누르냐에 따라 출력되는 숫자가 달라진다. 예를 들어, 계산기를 작동시키면 1이 출력되고, 여기서 Mul를 누르면 2가 되며, 또 Mul을 누르면 4가 출력된다.
영수는 이 계산기가 모든 숫자를 출력할 수 있는지 궁금해졌다. 하지만 영수는 버튼을 누르는 것 조차 매우 귀찮은 일이기 때문에, 계산기의 버튼을 최대한 적은 횟수만큼만 누르고 싶어 한다. 숫자 N이 주어질 때, 계산기를 작동시켜 숫자 N을 만들기 위하여 최소 몇 번 버튼을 눌러야 하는지를 구하는 프로그램을 작성하시오. 예를 들어, 숫자 10을 만들기 위해서는 1 → 2 → 4 → 8 → 16 → 5 → 10 가 되게끔 버튼을 누르면 되므로, 버튼을 6번만 누르면 되고, 이것이 최솟값이다. 참고로, 이상한 계산기는 5자리 정수까지밖에 표현할 수 없으며, 만약 연산 결과가 5자리를 넘어가는 경우에는 계산기가 고장나버린다. 따라서 항상 계산 결과를 최대 5자리로 유지해야 한다.
입력
첫째 줄에는 이상한 계산기로 만들고자 하는 숫자 N이 주어진다. ( 1 ≤ N ≤ 99,999 ) 단, 계산기로 만들 수 없는 값은 주어지지 않는다.
출력
이상한 계산기로 숫자 N을 만들기 위해 눌러야 하는 최소 버튼 클릭 횟수를 출력한다.
예제 입력
10
예제 출력
6
2. 풀이
이 문제는 BFS로 해결이 가능하다.
예를 들어 5를 찾는다고 가정해보면, 다음과 같은 순서로 진행이 된다.
- 0 레벨
1을 찾는 최소 횟수는 0
- 1 레벨
0을 찾는 최소 횟수는 1
2를 찾는 최소 횟수는 1
- 2 레벨
0을 찾는 횟수는 2가 되지만, 0을 찾는 횟수가 레벨 1일 때가 가장 최소 횟수이기 때문에 0을 찾는 최소 횟수는 여전히 1
4를 찾는 최소 횟수는 2
- 3 레벨
1을 찾는 횟수는 3이 되지만, 1을 찾는 횟수가 레벨 0일 때가 가장 최소 횟수이기 때문에 1을 찾는 최소 횟수는 여전히 0
8을 찾는 최소 횟수는 3
- 4 레벨
2를 찾는 횟수는 4가 되지만, 2를 찾는 횟수가 레벨 1일 때가 가장 최소 횟수이기 때문에 2를 찾는 최소 횟수는 여전히 1
16을 찾는 최소 횟수는 4
- 5 레벨
5를 찾는 최소 횟수는 5
32를 찾는 최소 횟수는 5
이를 코드에 적용하면 다음과 같다.
import java.io.*;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.Queue;
public class Main {
private static final BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
private static final BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));
private static int n;
private static int[] level;
private static int count;
public static void main(String[] args) throws IOException {
n = Integer.parseInt(br.readLine());
level = new int[1000000];
bfs(1);
bw.write(level[n] - 1 + "");
br.close();
bw.flush();
bw.close();
}
private static void bfs(int v) {
Queue<Integer> queue = new LinkedList<>();
queue.add(v);
level[v] = v;
while(!queue.isEmpty()) {
int w = queue.poll();
int mul = w * 2;
int div = w / 3;
if(mul > 0 && mul <= 99999) {
if(level[mul] == 0) {
queue.add(mul);
level[mul] = level[w] + 1;
} else {
level[mul] = Math.min(level[mul], level[w] + 1);
}
if(mul == n) break;
}
if(div > 0 && div <= 99999) {
if(level[div] == 0) {
queue.add(div);
level[div] = level[w] + 1;
} else {
level[div] = Math.min(level[div], level[w] + 1);
}
if(div == n) break;
}
}
}
}
'코딩 테스트 > 실전 문제' 카테고리의 다른 글
[실전 문제] 폭발물 설치 (0) | 2021.03.04 |
---|---|
[실전 문제] 파티 (0) | 2021.03.04 |
[실전 문제] 2색 칠하기 (0) | 2021.02.27 |
[실전 문제] 팰린드롬 만들기 (0) | 2021.02.27 |