[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의 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.