오늘의 인기 글
최근 글
최근 댓글
Today
Total
12-21 17:24
관리 메뉴

우노

[DP] 이코테 “못생긴 수” Python 풀이 본문

Algorithm/Dynamic Programming

[DP] 이코테 “못생긴 수” Python 풀이

운호(Noah) 2022. 9. 21. 15:01

문제

  • 못생긴 수란 오직 2, 3, 5 만을 소인수로 가지는 수를 의미한다.
  • 다시 말해 오직 2, 3, 5를 약수로 가지는 합성수를 의미한다.
    • 12의 약수인 1, 2, 3, 4, 6, 12 중에서 2와 3이 소인수이다.
  • 1은 못생긴 수라고 가정한다.
  • 따라서 못생긴 수들은 {1, 2, 3, 4, 5, 6, 8, 9, 10, 12, 15, ... } 순으로 이어지게 된다.
  • 이 때, n번째 못생긴 수를 찾는 프로그램을 작성해라.
  • 예를 들어 11번째 못생긴 수는 10이다.

입력 조건

  • 첫째 줄에 n이 입력된다.(1 <= n <= 1,000)

출력 조건

  • n번째 못생긴 수를 출력한다.

입력 예시

10
4

출력 예시

12
4

풀이

  • 2의 배수 변수, 3의 배수 변수, 5의 배수 변수에 대하여 각각 ‘가장 작은 못생긴 수’부터 오름차순으로 하나씩 확인하면서,
  • 각 배수를 곱한 값도 ‘못생긴 수’가 될 수 있도록 처리한다.
  • 예를 들어, 먼저 못생긴 수로 1이 있다고 해보자.
  • 이때 각각 2의 배수, 3의 배수, 5의 배수를 구하면 다음과 같다.
    • 2의 배수 : 1 x 2 = 2
    • 3의 배수 : 1 x 3 = 3
    • 5의 배수 : 1 x 5 = 5
  • 이로써 우리는 새롭게 2, 3, 5 또한 못생긴 수에 해당한다는 것을 알 수 있다.
  • 따라서, 전체 못생긴 수는 {1, 2, 3, 5}가 된다.
  • 첫 번째로 못생긴 수인 1에 이어서 그다음으로 못생긴 수는 2가 된다.
  • 이때 각각 2의 배수, 3의 배수, 5의 배수를 구하면 다음과 같다.
    • 2의 배수 : 2 x 2 = 4
    • 3의 배수 : 2 x 3 = 6
    • 5의 배수 : 2 x 5 = 10
  • 이로써 우리는 4, 6, 10이 못생긴 수에 해당한다는 것을 알 수 있다.
  • 따라서, 전체 못생긴 수는 {1, 2, 3, 4, 6, 10}이 된다.
  • 이렇게 못생긴 수들을 작은 수부터 차례대로 확인하면서,
  • 각 못생긴 수에 대해서 2의 배수 3의 배수, 5의 배수를 고려하며 풀면 된다.

예제 코드

import sys

# n번째 못생긴 수 출력
n = int(sys.stdin.readline())

ugly = [0] * n # 못생긴 수를 담기 위한 테이블 (1차원 DP 테이블)
ugly[0] = 1 # 첫 번째 못생긴 수는 1

# 몇 번쨰 인덱스까지 2, 3, 5배한 결과를 다뤘는지
i2 = i3 = i5 = 0

# 다음으로 다뤄야하는 곱셈 값
next2, next3, next5 = 2, 3, 5

# 1부터 n까지의 못생긴 수들을 찾기
for l in range(1, n):

    # 다음으로 다뤄야하는 곱셈 값들 중 최솟값을 l번째 인덱스에 저장
    ugly[l] = min(next2, next3, next5)

    # 현재 i2 인덱스 값에 2배를 곱한 값은 업데이트 되었으므로, 
    # 다음 i2 인덱스 값에 2배를 곱한 값이 무엇인지 임시 저장
    if ugly[l] == next2:
        i2 += 1
        next2 = ugly[i2] * 2

    # 현재 i3 인덱스 값에 3배를 곱한 값은 업데이트 되었으므로, 
    # 다음 i3 인덱스 값에 3배를 곱한 값이 무엇인지 임시 저장
    if ugly[l] == next3:
        i3 += 1
        next3 = ugly[i3] * 3

    # 현재 i5 인덱스 값에 5배를 곱한 값은 업데이트 되었으므로, 
    # 다음 i5 인덱스 값에 5배를 곱한 값이 무엇인지 임시 저장
    if ugly[l] == next5:
        i5 += 1
        next5 = ugly[i5] * 5

# n번째 못생긴 수를 출력
print(ugly[n - 1])

참고

  • 이것이 취업을 위한 코딩 테스트다. with python
Comments