BOJ 20301 - 반전 요세푸스 (Python)

조민수·2024년 2월 14일
0

BOJ

목록 보기
8/64

S3, 구현, 덱


문제 설명

요세푸스 문제는 다음과 같다.

11번 사람 오른쪽에는 22번 사람이 앉아 있고, 22번 사람 오른쪽에는 33번 사람이 앉아 있고,
계속하여 같은 방식으로 NN명의 사람들이 원을 이루며 앉아 있다.
NN번 사람 오른쪽에는 11번 사람이 앉아 있다.
이제 KK(N\leq N)번 사람을 우선 제거하고, 이후 직전 제거된 사람의 오른쪽의 KK번째 사람을 계속 제거해 나간다.
모든 사람이 제거되었을 때, 제거된 사람의 순서는 어떻게 될까?

이 문제의 답을 (N\boldsymbol{N}, K\boldsymbol{K})–요세푸스 순열이라고 하며, (77, 33)–요세푸스 순열은 <3,6,2,7,5,1,4>\left<3, 6, 2, 7, 5, 1, 4\right>가 된다.

하지만 한 방향으로만 계속 돌아가는 건 너무 지루하다. 따라서 요세푸스 문제에 재미를 더하기 위해
MM명의 사람이 제거될 때마다 원을 돌리는 방향을 계속해서 바꾸려고 한다. 이렇게 정의된 새로운 문제의 답을 (N\boldsymbol{N}, K\boldsymbol{K}, M\boldsymbol{M})–반전 요세푸스 순열이라고 하며, (77, 33, 44)–반전 요세푸스 순열은 <3,6,2,7,1,5,4>\left<3, 6, 2, 7, \mathbf{1}, \mathbf{5}, \mathbf{4}\right>가 된다.

NN, KK, MM이 주어질 때, (NN, KK, MM)–반전 요세푸스 순열을 계산해 보자.

입력

첫째 줄에 정수 NN, KK, MM이 주어진다.

  • (1N5 0001 \leq N \leq 5\ 000, 1K,MN1 \leq K, M \leq N)

출력

(NN, KK, MM)–반전 요세푸스 순열을 이루는 수들을 한 줄에 하나씩 순서대로 출력한다.

입력 예시

7 3 4

출력 예시

3
6
2
7
1
5
4

풀이

deque을 사용하지 않은 풀이

  • idx를 k 만큼 올리고 내리는데 어려움이 있었다.
n, k, m = map(int, input().split())

q = [i for i in range(1, n + 1)]

idx = 0
cnt = 0

flag = False        # False면,-> | True면, <-
while len(q) > 0:
    if cnt % m == 0:
        flag = not flag

    if flag:
        idx = (idx + k - 1) % len(q)
    else:
        idx = (idx - k) % len(q)

        while idx < 0:
            idx += len(q)
    cnt += 1
    print(q[idx])
    q.pop(idx)

힌트를 참고한 deque를 이용한 풀이

  • 오른쪽으로 돌 때는 k-1번, 왼쪽으로 돌 때는 k번 돈다는 점을 유의해야 한다.
from collections import deque

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

q = deque(range(1, n + 1))
idx = 0

while q:
    if idx // m % 2 == 0:
        for _ in range(k - 1):
            q.append(q.popleft())
    else:
        for _ in range(k):
            q.appendleft(q.pop())
    idx += 1
    print(q.popleft())
profile
사람을 좋아하는 Front-End 개발자

0개의 댓글