Study/Baekjoon
[Java] Baekjoon1158: 요세푸스 문제
devyoseph
2022. 2. 11. 10:50
요세푸스 문제
2 초 | 256 MB | 58856 | 28719 | 20405 | 48.052% |
문제
요세푸스 문제는 다음과 같다.
1번부터 N번까지 N명의 사람이 원을 이루면서 앉아있고, 양의 정수 K(≤ N)가 주어진다. 이제 순서대로 K번째 사람을 제거한다. 한 사람이 제거되면 남은 사람들로 이루어진 원을 따라 이 과정을 계속해 나간다. 이 과정은 N명의 사람이 모두 제거될 때까지 계속된다. 원에서 사람들이 제거되는 순서를 (N, K)-요세푸스 순열이라고 한다. 예를 들어 (7, 3)-요세푸스 순열은 <3, 6, 2, 7, 5, 1, 4>이다.
N과 K가 주어지면 (N, K)-요세푸스 순열을 구하는 프로그램을 작성하시오.
입력
첫째 줄에 N과 K가 빈 칸을 사이에 두고 순서대로 주어진다. (1 ≤ K ≤ N ≤ 5,000)
출력
예제와 같이 요세푸스 순열을 출력한다.
풀이
요세푸스 순열은 큐를 활용하는 대표적인 문제 중 하나입니다.
큐에 1부터 N까지 숫자들을 집어넣고 반복문을 통해 큐에 아무 원소도 없을 때까지 시행을 반복해줍니다.
원의 특성
원은 처음과 끝이 연결되어있습니다.
이를 큐에 원소를 집어넣고 처음 원소를 끝에 다시 집어넣는 방식으로 구현할 수 있습니다.
1 2 3 4 → 2 3 4 1 → 3 4 1 2 → 4 1 2 3 → 1 2 3 4
K번째마다 꺼내기
while 문과 Queue 클래스의 isEmpty() 를 조합해 큐가 텅 빌 때까지 반복문을 돌릴 수 있습니다.
while(!Q.isEmpty()) {
Q.offer(Q.poll()); // 원소를 빼고 뒤로 넘김
}
외부에서 변수를 하나 만들고 반복문이 돌아갈 때마다 더해줍니다.
이 변수가 입력 받은 K의 배수가될 때마다 큐의 원소를 꺼내고 기록해줍니다.
전체 코드
import java.util.*;
public class Main {
public static void main(String[] args) {
Queue<Integer> Q = new LinkedList<Integer>();
StringBuilder sb = new StringBuilder();
Scanner sc = new Scanner(System.in);
sb.append("<");
int N = sc.nextInt();
int K = sc.nextInt();
for(int i=1; i<=N; i++) { //큐에 1부터 N까지 넣기
Q.offer(i);
}
int idx = 0;
while(!Q.isEmpty()) {
idx ++; // idx로 실시간 순번을 기록
if(idx%K==0) { // idx가 K의 배수가 될 때 해당 원소를 꺼내고 기록
sb.append(Q.poll()+", ");
}else {
Q.offer(Q.poll()); // 그 외에는 빼고 뒤로 넘김
}
}
sb.setLength(sb.length()-2); // 마지막에 더했던 ', ' 를 삭제
sb.append(">");
System.out.println(sb.toString());
}
}