[Day4] [코딩테스트 연습] 동적계획법(Dynamic Promgramming) - N으로표현

승준·2021년 4월 24일
0

[코딩테스트 연습] 동적계획법(Dynamic Promgramming) - N으로표현

문제 설명

아래와 같이 5와 사칙연산만으로 12를 표현할 수 있습니다.

12 = 5 + 5 + (5 / 5) + (5 / 5)
12 = 55 / 5 + 5 / 5
12 = (55 + 5) / 5

5를 사용한 횟수는 각각 6,5,4 입니다. 그리고 이중 가장 작은 경우는 4입니다.
이처럼 숫자 N과 number가 주어질 때, N과 사칙연산만 사용해서 표현 할 수 있는 방법 중 N 사용횟수의 최솟값을 return 하도록 solution 함수를 작성하세요.

제한사항
  • N은 1 이상 9 이하입니다.
  • number는 1 이상 32,000 이하입니다.
  • 수식에는 괄호와 사칙연산만 가능하며 나누기 연산에서 나머지는 무시합니다.
  • 최솟값이 8보다 크면 -1을 return 합니다.
입출력 예
Nnumberreturn
5124
2113
입출력 예 설명

예제 #1
문제에 나온 예와 같습니다.

예제 #2
11 = 22 / 2와 같이 2를 3번만 사용하여 표현할 수 있습니다.

동적계획법 (Dyanamic Programming)

주어진 최적화 문제를 재귀적인 방식으로 보다 작은 부분으로 나누어 부분 문제로 풀어, 이 해를 조합 하여 전체 문제의 해답을 구하는 방식

알고리즘에 진행에 따라 탐색해야 할 범위를 동적으로 결정 함으로써 탐색 범위를 한정할 수 있음

적용 사례

피보나치 수열

재귀 함수

피보나치 수열을 재귀 함수로 구현 한다면? 복잡도가 지수함수형태를 띄게 된다

image-20210424112014308

동적 계획법

부분 문제(f(0)=0, f(1)=1)로 풀어 이들을 조합 한다. 복잡도는 선형함수의 형태로 나타난다.

image-20210424112125554

Knapsack 문제

가장 높은 값을 가지도록 물건을 골라 배낭에 담으시오. 배낭 문제 - 위키백과, 우리 모두의 백과사전

문제의 해결

동적계획법으로 설계

N을 한 번 사용해서 만들수 있는 수(들) -> 1

N을 두 번 사용해서 만들수 있는 수(들) -> 2

N을 세 번 사용해서 만들수 있는 수(들) -> 3

이를 반복하다가 내가 원하는 수가 나타나면 이를 반환 한다.

예제

N = 5 일때, 1번 사용해서는 1개의 숫자만 만들 수 있다. 2번 사용 했을 때는, 551개를 사용해서 만들 것을 사칙 연산한 것으로 각각 만들 수 있다.

image-20210424112826721

image-20210424113053357

image-20210424113122357

이를 일반화 하면 다음과 같다.

image-20210424113347536

코드 구현

# N이 1~ 9까지
dp = [set() for _ in range(1,9)]

# i개 만큼 만들수 있는 숫자 추가
for i ,x in enumerate(dp, start=1):
    x.add(int(str(N)*i))


for i in range(1, len(dp)):
    # i=1,
    for j in range(i):
        for op1 in dp[j]:
            for op2 in dp[i-j-1]:
                dp[i].add(op1+op2)
                dp[i].add(op1-op2)
                dp[i].add(op1*op2)
                if op2 != 0:
                    dp[i].add(op1//op2)

    if number in dp[i]:
        answer = i + 1
        break
else:
    answer = -1


print(answer)

더 나은 풀이

op1op2를 구하는데 itertoolsproduct 메소드를 사용해서 구현 하였다.

from itertools import product

def solution(N, number):
    dp = [set() for _ in range(8+1)]

    for i in range(1, 8+1):
        dp[i].add(int(str(N)*i))
        for j in range(1, i):
            for op1, op2 in product(dp[j], dp[i-j]):
                dp[i].add(op1+op2)
                dp[i].add(op1-op2)
                dp[i].add(op1*op2)
                if op2 != 0:
                    dp[i].add(op1//op2)
        if number in dp[i]:
            return  (i)

    return -1

기록 사항

  • enumerate(iterable, start=0)
    • enumerate 함수에 start인자가 있는지 처음 알았다. start를 쓰지 못해서 어려운적이 많았는데 꼭 기억하고 다음 부터 써야 겠다
  • dict.add(), dict.get()
    • dict에 다양한 메소드가 있는 걸 을 배웠다. 보통 리스트를 많이 사용해서 append 메소드만 원소 추가가 가능 한 줄 알 고있었는데 add() 로도 가능하고, 추가적으로 get() 메소드를 사용해서 key에 접근할 수 있다. 이때 디폴트 값을 설정 해주면 value가 없을때 디폴트 값을 가져오고 디폴트를 설정 안해주면 None을 반환 한다.
profile
내일을 기록하기 위해서 오늘을 기록합니다 🤗

0개의 댓글