1547. Minimum Cost to Cut a Stick
Problem Description
The problem presents the scenario where you have a wooden stick of length n
units, marked at each unit along its length. For example, a stick that is 6 units long would be marked at 0, 1, 2, 3, 4, 5, and 6. You are given an integer array cuts
where cuts[i]
represents a position along the stick where you need to make a cut.
The primary task is to cut the stick at these marked positions in such a way that the total cost is minimized. The cost of making a cut is equal to the current length of the stick being cut. After a cut, the stick is divided into two smaller sticks. The objective is to determine the minimum total cost to perform all the cuts.
A key point to note is that the order of the cuts can be changed. The goal is not just to perform the cuts in the order they are given, but to find the order that results in the least total cost.
Intuition
To arrive at the solution, we need to realize that this problem is a classic example of dynamic programming, specifically the matrix-chain multiplication problem. The main idea is to find the most cost-effective order to perform the cuts, akin to finding the optimal parenthesization of a product of matrices.
Here's the intuition to approach the problem:
- Introduce "virtual" cuts at the start and end of the stick (positions 0 and
n
). These do not contribute to the cost but serve as the boundaries of the stick. - Arrange the cuts in a sorted sequence, including the virtual cuts.
- Define a recursive relation to represent the minimum cost of making cuts between two endpoints on the stick.
- Realize that the best way to split the stick involves splitting it at some point
k
between two endpoints, leading to two new problems of the same nature but on smaller sticks. The cost to make a cut atk
would include the cost of cutting the left part, the cost of cutting the right part, and the cost of the cut itself, which is the length of the stick between the two endpoints. - The dynamic programming aspect involves storing solutions to subproblems to avoid redundant calculations. This is done by filling a table
f
with the minimum costs for cutting between every possible pair of cuts. - The final solution will be the computed value representing the minimum cost to cut the entire stick between the first and last positions (cuts[0] and cuts[-1]).
To ensure we don't miss the best split point for a given segment of the stick, every possible split point k
needs to be considered, and the minimum cost to cut between the endpoints will be stored in the table f
.
Learn more about Dynamic Programming and Sorting patterns.
Solution Approach
The implementation of the solution follows the dynamic programming approach, and here is a step-by-step explanation:
-
Include the start and end of the stick as cuts, so we add
0
andn
to the list of cuts for convenience. After this, thecuts
array contains all the positions where we need to make the cuts. Inserting the beginning and end of the stick into our cuts array doesn't change the problem but simplifies the implementation.1cuts.extend([0, n])
-
Sort the
cuts
array. It's important that thecuts
array is sorted so that we can consider each segment between two cuts in order. This array now acts as a reference for our dynamic programming.1cuts.sort()
-
Create a 2D list
f
for storing the minimum costs with all values initially set to zero. The size of the list is the number of cuts including the virtual cuts at the beginning and the end.f[i][j]
will store the minimum cost to cut the stick fromcuts[i]
tocuts[j]
.1m = len(cuts) 2f = [[0] * m for _ in range(m)]
-
Use a nested loop to consider all segments between cuts with length
l
ranging from 2 tom
(inclusive), effectively covering all possible sub-segments of the stick.l=2
means just one cut in the segment (since we're including the two virtual cuts at the ends), andl=m
would mean considering the entire stick.1for l in range(2, m): 2 for i in range(m - l): 3 j = i + l 4 f[i][j] = inf 5 ...
-
For each pair of indices
i
andj
, iterate through all potential cutting pointsk
betweeni
andj
to find the minimum cost. We have initializedf[i][j]
with infinity, and we aim to find the least possible cost.1for k in range(i + 1, j): 2 f[i][j] = min(f[i][j], f[i][k] + f[k][j] + cuts[j] - cuts[i])
-
The above step considers each possible way to split the stick into two parts: the left part from
cuts[i]
tocuts[k]
and the right part fromcuts[k]
tocuts[j]
. The minimum cost for the split positionk
is the sum of the minimum cost to cut the left partf[i][k]
, the minimum cost to cut the right partf[k][j]
, and the cost of making the actual cut atk
which iscuts[j] - cuts[i]
(the current length of the stick segment we're cutting). -
Finally, return the value
f[0][-1]
which now contains the minimum cost to cut the entire stick, taking into the account the order in which cuts must be made to minimize the total cost.1return f[0][-1]
The implementation uses a dynamic programming table to ensure that the solution to each subproblem is only calculated once and then reused, leading to a more efficient algorithm than a naive recursive approach would allow.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's walk through a small example to illustrate the solution approach.
Suppose we have a wooden stick with a length of n = 7
units, and we have been asked to make cuts at positions given by the array cuts = [1, 3, 4, 5]
.
-
First, we include the start and end of the stick as cuts according to step 1, so our cuts array becomes
cuts = [1, 3, 4, 5, 0, 7]
. -
Next, we sort the
cuts
array, resulting incuts = [0, 1, 3, 4, 5, 7]
. This allows us to treat this array as a sequence of segments to consider in order. -
Following step 3, we create a 2D list
f
for storing the minimum costs. Since there are6
positions to make cuts, including the virtual ones at the beginning and at the end, our matrix will be 6 by 6, filled with zeros. -
Now we consider all segments between cuts with lengths ranging from 2 to 6 (the size of our
cuts
array). -
For a segment length of 2 (meaning a single cut between two points), the cost is straightforward. It is the difference between the points since there's no choice in how to split the stick. For example:
- For segment
[0, 1, 3]
, the cost of cutting the stick at point 1 is the difference,3 - 0 = 3
. - However, for segments longer than that, we must consider potential splitting points. Let's take a segment
[0, 1, 3, 4]
. We can cut at1
or at3
. - If we cut at
1
first, the cost is4 (length of the segment) + 2 (for cutting the second piece at 3) = 6
. - If we cut at
3
first, the cost is4 (length of the segment) + 1 (for cutting the first piece at 1) = 5
. So we'll choose to cut at3
first, and the minimum cost would be stored inf[0][3]
.
- For segment
-
We continue this process for larger segments, calculating costs and choosing the minimum until the entire matrix
f
is filled. For each segment, the selection of where to cut depends on the previously calculated minimum costs for smaller segments. This cumulative buildup ensures that we are using optimal substructure to find the minimum cost. -
After we finish populating our matrix, the value stored at
f[0][5]
(which corresponds to the full length of the stick fromcuts[0]
tocuts[-1]
) will be the minimum cost to cut the entire stick.
Let's simulate this for our segments. We assume inf
is a placeholder for infinity.
0 | 1 | 3 | 4 | 5 | 7 | |
---|---|---|---|---|---|---|
0 | 0 | 0 | 3 | 4 | 5 | 8 |
1 | 0 | 0 | 3 | 4 | 5 | |
3 | 0 | 0 | 3 | 4 | ||
4 | 0 | 0 | 3 | |||
5 | 0 | 0 |
- The cost of making a cut at the last segment (from 5 to 7) is the length of the stick from 5 to 7, which is 2 units. This is reflected in
f[4][5]
at the end of the matrix. - As observed during the example, the cost to cut at position
3
when considering from0
to4
is less than cutting at position1
, sof[0][3]
is populated with5
. - The final cost
f[0][5]
indicates that the entire stick can be cut at8
units, which is the minimum cost to perform the cuts at[1, 3, 4, 5]
.
By systematically considering each possible segment and sub-segment, storing the costs, and building upwards from those, we have used dynamic programming to avoid redundant calculations and to find the minimum cost to cut the stick at specific points.
Solution Implementation
1class Solution:
2 def minCost(self, n: int, cuts: List[int]) -> int:
3 # Extend the list of cuts to include the start (0) and the end (n)
4 cuts.extend([0, n])
5 # Sort the cuts so they are in ascending order
6 cuts.sort()
7 # Count the total number of cuts (including start and end)
8 num_cuts = len(cuts)
9 # Create a 2D list (matrix) filled with zeros for dynamic programming
10 dp = [[0] * num_cuts for _ in range(num_cuts)]
11
12 # Start evaluating the minimum cost in increasing order of lengths
13 for length in range(2, num_cuts):
14 # Iterate over the cuts for the current length
15 for start in range(num_cuts - length):
16 end = start + length
17 # Initialize current cell with infinity, representing the uncalculated minimum cost
18 dp[start][end] = float('inf')
19
20 # Check for the best place to split the current piece being considered
21 for k in range(start + 1, end):
22 # Calculate the cost of the current cut and compare it
23 # to the cost already present in this cell
24 dp[start][end] = min(dp[start][end],
25 dp[start][k] + dp[k][end] + cuts[end] - cuts[start])
26
27 # Return the minimum cost to make all cuts for the full length of the rod
28 return dp[0][-1]
29
1class Solution {
2 public int minCost(int n, int[] cuts) {
3 // Initialize a new list to hold the cuts as well as the two ends of the stick
4 List<Integer> cutPoints = new ArrayList<>();
5
6 // Add each cut point from the given cuts array to the list
7 for (int cut : cuts) {
8 cutPoints.add(cut);
9 }
10
11 // Add the start and the end of the stick to the list as cut points
12 cutPoints.add(0);
13 cutPoints.add(n);
14
15 // Sort the list of cut points (includes the ends of the stick now)
16 Collections.sort(cutPoints);
17
18 // Calculate the size of the list post additions
19 int size = cutPoints.size();
20
21 // Initialize a 2D array to store the minimum cost for cutting between two points
22 int[][] dp = new int[size][size];
23
24 // Start from the second cut point because the first will always be zero
25 for (int length = 2; length < size; ++length) {
26 // Determine the minimum cost for each range of cut points
27 for (int i = 0; i + length < size; ++i) {
28 // j marks the end of the range starting from i
29 int j = i + length;
30
31 // Initialize the minimum cost for this range as a large number
32 dp[i][j] = Integer.MAX_VALUE;
33
34 // Try all possible intermediate cuts to find the minimum cost split
35 for (int k = i + 1; k < j; ++k) {
36 // Calculate the cost of making this cut and if it's the new minimum, update dp[i][j]
37 dp[i][j] = Math.min(dp[i][j], dp[i][k] + dp[k][j] + cutPoints.get(j) - cutPoints.get(i));
38 }
39 }
40 }
41
42 // Return the minimum cost to cut the entire stick
43 return dp[0][size - 1];
44 }
45}
46
1#include <vector>
2#include <algorithm>
3using namespace std;
4
5class Solution {
6public:
7 int minCost(int totalLength, vector<int>& cuts) {
8 // Include the edges as cuts (start and end of the rod)
9 cuts.push_back(0);
10 cuts.push_back(totalLength);
11 // Sort the cuts for dynamic programming approach
12 sort(cuts.begin(), cuts.end());
13
14 int numOfCuts = cuts.size();
15 int costMatrix[110][110] = {0}; // Assuming we will not have more than 100 cuts
16
17 // Filling cost matrix diagonally, starting from the smallest subproblems
18 for (int len = 2; len < numOfCuts; ++len) {
19 for (int i = 0; i + len < numOfCuts; ++i) {
20 int j = i + len;
21 costMatrix[i][j] = 1 << 30; // Initialize with a large number (infinity)
22
23 // Find the minimum cost to cut this piece further
24 for (int k = i + 1; k < j; ++k) {
25 int cost = costMatrix[i][k] + costMatrix[k][j] + cuts[j] - cuts[i];
26 costMatrix[i][j] = min(costMatrix[i][j], cost);
27 }
28 }
29 }
30
31 // Return the minimum cost to cut the whole rod
32 return costMatrix[0][numOfCuts - 1];
33 }
34};
35
1function minCost(n: number, cuts: number[]): number {
2 // Add the start and end of the rod to the cuts array
3 cuts.push(0);
4 cuts.push(n);
5
6 // Sort the cuts in ascending order
7 cuts.sort((a, b) => a - b);
8
9 // Determine the number of cuts including the start and end positions
10 const cutCount = cuts.length;
11
12 // Initialize a 2D array (matrix) to store the minimum cost for each segment
13 const dp: number[][] = new Array(cutCount).fill(0).map(() => new Array(cutCount).fill(0));
14
15 // Iterate over the lengths of the segments starting from 2 (since we need at least two cuts to define a segment)
16 for (let len = 2; len < cutCount; ++len) {
17 // Iterate over all possible segments for the current length
18 for (let i = 0; i + len < cutCount; ++i) {
19 // Calculate the end index of the segment
20 const j = i + len;
21
22 // Initialize the minimum cost for current segment to a large number (acts like infinity)
23 dp[i][j] = Number.MAX_SAFE_INTEGER;
24
25 // Iterate over all possible positions to make a cut within the current segment
26 for (let k = i + 1; k < j; ++k) {
27 // Calculate the cost of cutting at position k and add the cost of the two resulting segments
28 // Update dp[i][j] with the minimum cost found so far
29 dp[i][j] = Math.min(dp[i][j], dp[i][k] + dp[k][j] + cuts[j] - cuts[i]);
30 }
31 }
32 }
33
34 // Return the minimum cost to cut the rod into the specified pieces
35 return dp[0][cutCount - 1];
36}
37
Time and Space Complexity
Time Complexity
The time complexity of the provided code can be analyzed by examining the nested loops and the operations within those loops:
-
There are two outer loops iterating over the length
l
from 2 tom
(wherem
is the total number of cuts plus the two endpoints, 0 andn
), and over the starting indexi
. This results in a time complexity of approximatelyO(m^2)
for these two loops. -
Inside the two outer loops, there is an inner loop that iterates over the possible intermediate cuts
k
between thei
th cut and thej
th cut. In the worst case, this inner loop runsO(m)
times. -
The operations inside the inner loop involve computing the minimum of the current value and the sum of the costs for the left and right pieces plus the cost of making this cut:
f[i][k] + f[k][j] + cuts[j] - cuts[i]
. This operation runs in constant time,O(1)
.
Multiplying these together, the overall time complexity is O(m^2) * O(m) * O(1) = O(m^3)
.
Space Complexity
The space complexity can be determined by analyzing the memory allocations in the code:
-
The 2D list
f
of sizem x m
is the primary data structure, wherem
is the number of cuts plus 2 (for the endpoints). This 2D list constitutes a space complexity ofO(m^2)
. -
No other significant data structures are used, and the input list
cuts
is modified in place (not requiring additional space relative to inputs).
Thus, the space complexity is O(m^2)
.
Learn more about how to find time and space complexity quickly using problem constraints.
What's the output of running the following function using input 56
?
1KEYBOARD = {
2 '2': 'abc',
3 '3': 'def',
4 '4': 'ghi',
5 '5': 'jkl',
6 '6': 'mno',
7 '7': 'pqrs',
8 '8': 'tuv',
9 '9': 'wxyz',
10}
11
12def letter_combinations_of_phone_number(digits):
13 def dfs(path, res):
14 if len(path) == len(digits):
15 res.append(''.join(path))
16 return
17
18 next_number = digits[len(path)]
19 for letter in KEYBOARD[next_number]:
20 path.append(letter)
21 dfs(path, res)
22 path.pop()
23
24 res = []
25 dfs([], res)
26 return res
27
1private static final Map<Character, char[]> KEYBOARD = Map.of(
2 '2', "abc".toCharArray(),
3 '3', "def".toCharArray(),
4 '4', "ghi".toCharArray(),
5 '5', "jkl".toCharArray(),
6 '6', "mno".toCharArray(),
7 '7', "pqrs".toCharArray(),
8 '8', "tuv".toCharArray(),
9 '9', "wxyz".toCharArray()
10);
11
12public static List<String> letterCombinationsOfPhoneNumber(String digits) {
13 List<String> res = new ArrayList<>();
14 dfs(new StringBuilder(), res, digits.toCharArray());
15 return res;
16}
17
18private static void dfs(StringBuilder path, List<String> res, char[] digits) {
19 if (path.length() == digits.length) {
20 res.add(path.toString());
21 return;
22 }
23 char next_digit = digits[path.length()];
24 for (char letter : KEYBOARD.get(next_digit)) {
25 path.append(letter);
26 dfs(path, res, digits);
27 path.deleteCharAt(path.length() - 1);
28 }
29}
30
1const KEYBOARD = {
2 '2': 'abc',
3 '3': 'def',
4 '4': 'ghi',
5 '5': 'jkl',
6 '6': 'mno',
7 '7': 'pqrs',
8 '8': 'tuv',
9 '9': 'wxyz',
10}
11
12function letter_combinations_of_phone_number(digits) {
13 let res = [];
14 dfs(digits, [], res);
15 return res;
16}
17
18function dfs(digits, path, res) {
19 if (path.length === digits.length) {
20 res.push(path.join(''));
21 return;
22 }
23 let next_number = digits.charAt(path.length);
24 for (let letter of KEYBOARD[next_number]) {
25 path.push(letter);
26 dfs(digits, path, res);
27 path.pop();
28 }
29}
30
Recommended Readings
What is Dynamic Programming Prerequisite DFS problems dfs_intro Backtracking problems backtracking Memoization problems memoization_intro Pruning problems backtracking_pruning Dynamic programming is an algorithmic optimization technique that breaks down a complicated problem into smaller overlapping sub problems in a recursive manner and uses solutions to the sub problems to construct a solution
Sorting Summary Comparisons We presented quite a few sorting algorithms and it is essential to know the advantages and disadvantages of each one The basic algorithms are easy to visualize and easy to learn for beginner programmers because of their simplicity As such they will suffice if you don't know any advanced
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Got a question?Β Ask the Monster AssistantΒ anything you don't understand.
Still not clear? Β SubmitΒ the part you don't understand to our editors. Or join ourΒ Discord and ask the community.