Post

[Algorithm] DP Problems Series #3: Leetcode #746. Min Cost Climbing Stairs

재귀함수, top-down dp, bottom-up dp, cache 4가지 방법으로 구현하기 #3

[Algorithm] DP Problems Series #3: Leetcode #746. Min Cost Climbing Stairs

DP Problems Series에서는 DP 문제를 접했을 때 풀이할 수 있는 4가지 방법(재귀함수, top-down dp, bottom-up dp, cache)으로 정리하고 풀어본다.


Intro

✏️ LeetCode #746. Min Cost Climbing Stairs

Leetcode의 Min Cost Climbing Stairs 문제를 4가지 방식(재귀, Top-Down DP, Bottom-Up DP, Cache)으로 풀어보자.


Solution 1) Recursive

Time Complexity: O(2n)

아래 코드에서는 클래스 Solution 내에 minCostClimbingStairs 메서드가 정의되어 있으며, 여기서 중첩함수 recur를 사용하여 재귀적으로 최소 비용을 계산한다.

1
2
3
4
5
6
7
8
9
10
11
class Solution:
    def minCostClimbingStairs(self, cost: list[int]) -> int:
        n = len(cost)

        def recur(n):
            if n <= 1:
                return 0

            return min(recur(n - 1) + cost[n - 1], recur(n - 2) + cost[n - 2])

        return recur(n)


이번엔 중첩함수를 쓰지 않고 메소드를 분리해서 코드를 짜보자!

1
2
3
4
5
6
7
8
9
10
11
12
class Solution:
    def minCostClimbingStairs(self, cost: list[int]) -> int:
        self.cost = cost
        n = len(cost)

        return self.recur(n)

    def recur(self, n: int) -> int:
        if n <= 1:
            return 0

        return min(self.recur(n - 1) + self.cost[n - 1], self.recur(n - 2) + self.cost[n - 2])

위 코드에서는 중첩함수 대신에 클래스의 또 다른 메서드인 recur를 분리하였다. 이 방식은 재귀 호출을 통해 동일한 계산을 수행하지만, recur 메서드가 클래스 내부의 다른 메서드로 분리되어 있어서 코드의 가독성과 구조가 개선되었다!


Solution 2) DP: Top-Down (memoization)

Time Complexity: O(n)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class Solution:
    def minCostClimbingStairs(self, cost: list[int]) -> int:
        self.cost = cost
        n = len(cost)

        return self.dp_top_down(n)

    def dp_top_down(self, n: int, dp: dict[int, int] | None = None) -> int:
        if dp is None:
            dp = {0: 0, 1: 0}

        if n not in dp:
            dp[n] = min(
                self.dp_top_down(n - 1) + self.cost[n - 1],
                self.dp_top_down(n - 2) + self.cost[n - 2],
            )

        return dp[n]


Solution 3) DP: Bottom-Up (for loop)

Time Complexity: O(n)

List 활용

1
2
3
4
5
6
7
8
9
10
11
12
13
14
class Solution:
    def minCostClimbingStairs(self, cost: list[int]) -> int:
        n = len(cost)

        if n <= 1:
            return 0

        dp = [0] * (n + 1)
        dp[0], dp[1] = 0, 0

        for idx in range(2, n + 1):
            dp[idx] = min(dp[idx - 1] + cost[idx - 1], dp[idx - 2] + cost[idx - 2])

        return dp[n]

Dictionary 활용

1
2
3
4
5
6
7
8
9
10
11
12
class Solution:
    def minCostClimbingStairs(self, cost: list[int]) -> int:
        n = len(cost)
        dp = {0: 0, 1: 0}

        if n <= 1:
            return dp[n]

        for key in range(2, n + 1):
            dp[key] = min(dp[key - 1] + cost[key - 1], dp[key - 2] + cost[key - 2])

        return dp[n]


Solution 4) functools.cache

Time Complexity: O(n)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
from functools import cache


@cache
class Solution:
    def minCostClimbingStairs(self, cost: list[int]) -> int:
        n = len(cost)

        def recur(n):
            if n <= 1:
                return 0

            return min(recur(n - 1) + cost[n - 1], recur(n - 2) + cost[n - 2])

        return recur(n)
This post is licensed under CC BY 4.0 by the author.