[Python] Leetcode 39. Combination Sum


Leetcode 39 Combination Sum

문제

Given an array of distinct integers candidates and a target integer target, return a list of all unique combinations of candidates where the chosen numbers sum to target. You may return the combinations in any order.

The same number may be chosen from candidates an unlimited number of times. Two combinations are unique if the frequency of at least one of the chosen numbers is different.

The test cases are generated such that the number of unique combinations that sum up to target is less than 150 combinations for the given input.

combination_sum_1

풀이

  • dfs로 풀 수 있음. 중복된 조합(combination)을 방지하기 위해 dfs인자에 start를 넣고 항상 start 이후부터 시작하도록 한다.

  • 왜 start 이후부터 시작하는지?
  • 예를 들어, [2, 3, 5] 조합으로 5을 만드는 방법을 생각해보자
    • 2를 사용한 모든 경우의 수를 체크했기 때문에, 3부터 시작할때는 2를 제외해도 됨.
candidats = [2, 3, 5]
target = 8

[]
    [2]
        [2, 2]
            [2, 2, 2]
                [2, 2, 2, 2] = 8
                [2, 2, 2, 3] = 9
                [2, 2, 2, 5] = 11
            [2, 2, 3]
                [2, 2, 3, 2] = 9
                [2, 2, 3, 3]  X
                [2, 2, 3, 5] X
            [2, 2, 5] X
        [2, 3]
            [2, 3, 2]
                [2, 3, 2, 2] X
                [2, 3, 2, 3] X
                [2, 3, 2, 5] X
            [2, 3, 3] O
            [2, 3, 5] X
        [2, 5]
            [2, 5, 2] X
            [2, 5, 3] X
            [2, 5, 5] X

    [3] -- 위에서 2가 들어간 모든 경우의 수를 계산했기 떄문에 2를 제외해도 됨.
        [3, 3]
            [3, 3, 3] X
            [3, 3, 5] X
        [3, 5]
            [3, 5, 3] X
            [3, 5] O
    [5] X
class Solution:
    def combinationSum(self, candidates: List[int], target: int) -> List[List[int]]:

        res = []
        def dfs(temp_sum, answer, start):
            if temp_sum > target:
                return
            elif temp_sum == target:
                res.append(answer)
                return
            for i in range(start, len(candidates)): ## start가 0이라면 2, 3, 5 셋다 쓸꺼고, start가 1이라면 3, 5만 쓴다는 뜻
                dfs(temp_sum+candidates[i], answer + [candidates[i]], i)
        dfs(0, [], 0)

        return res