2767. Partition String Into Minimum Beautiful Substrings
Problem Description
The goal is to split a given binary string s
into the minimum number of contiguous substrings where each substring satisfies two conditions:
- It does not start with a zero.
- It represents a number in binary that is a power of 5.
If this partition isn't possible according to the above rules, the output should be -1. This problem is an algorithmic challenge that requires identifying the specific partitions that ensure the minimum count and conform to the rules.
Intuition
To solve this problem, it's crucial to recognize that we're not simply looking for subsequences, but specifically for substrings (which cannot rearrange characters and must remain contiguous). Furthermore, these substrings need to be valid binary representations of powers of 5 without leading zeros.
The concept of dynamic programming may immediately come to mind to handle the optimization aspect of the problemâminimizing the number of substrings. Specifically, we use a depth-first search (DFS) with memoization to ensure efficient computation by avoiding redundant calculations.
Here's how we can approach the solution:
- Preprocess the binary equivalents of all possible powers of 5 that can be represented within the length of
s
and store them in a hash setss
. This preprocessing speeds up the checks needed later by allowing for constant-time lookups to see if a binary substring is a power of 5. - Implement the
dfs
function which operates recursively:- If at an index
i
, wheres[i]
is '0', we can immediately return infinity (inf
) because we can't have a leading zero. - If we reach the end of the string (
i >= n
), it means we have considered all characters, hence, no further partitions are needed, and we return 0. - Otherwise, we iterate from the current index
i
to the end of the string, treating each possible end indexj
as the end of a candidate substring. We calculate the decimal value of the substring as we extend it and check if that value is inss
. If it is, we have found a beautiful substring and we add 1 to the result ofdfs(j + 1)
.
- If at an index
- Optimize by using memoization to cache results of
dfs(i)
for each indexi
. This prevents the algorithm from re-computing the minimum number of substrings for any starting index more than once. - Call
dfs(0)
for the start of the string and if the value is infinite, we return -1 since it's impossible to partition the string into beautiful substrings. Otherwise, the value ofdfs(0)
gives us the minimum count of beautiful substrings.
This leads us to a recursive solution with memoization that efficiently computes the minimum required partitions by only considering valid power of 5 numbers and by avoiding redundant checks through memoization.
Learn more about Dynamic Programming and Backtracking patterns.
Solution Approach
The implementation starts with the creation of a set ss
which contains the binary representation of all the numbers that are powers of 5 up to the maximum length of the binary string s
provided. This set is pivotal for quickly verifying whether a substring can be considered beautiful.
A recursive function dfs(i)
is defined that takes an index i
and returns the minimum number of beautiful substrings starting from this index. The recursion uses two crucial base conditions:
- When the current position
i
is at the end of the string (i >= n
), which by definition means no further cuts are possible and it returns0
. - If the current bit is
0
, indicating a leading zero if it were to be the start of a substring, it returns infinity to signify this cannot form a beautiful substring.
The algorithm then progresses by considering all possible substrings starting from i
up to the end of the string. For each candidate substring, it performs the following steps:
- It uses bit shifting to calculate the binary to decimal conversion of the substring
s[i...j]
while iterating. - It checks if the current value, as it accumulates with each bit, exists in the precalculated powers of 5, stored in
ss
. If it does, it means the substring is beautiful. - It calls
dfs(j + 1)
for the remainder of the string starting fromj + 1
, wherej
is current end of the considered substring. To thisdfs
call, 1 is added representing the current beautiful substring just processed. - It continues to compute and track the minimum number of beautiful substrings (cuts) as it goes along.
The memoization is achieved using Python's @cache
decorator on the dfs
function. This optimization ensures that once a substring starting at index i
is processed, the result is stored and thus any subsequent calls to dfs(i)
will simply retrieve the stored result instead of recalculating, reducing the time complexity significantly.
At the end of recursion, if dfs(0)
returns infinity (inf
), it means the string s
cannot be divided into beautiful substrings and thus the function returns -1
. Otherwise, the minimum number of cuts will be returned.
To avoid repeatedly constructing the set ss
for powers of 5, it is precomputed once before the recursive calls. This is performed in a loop that starts with x = 1
and keeps multiplying x
by 5
, adding each new power of 5 into the ss
. This loop runs as long as the numbers being added are within the possible range dictated by the length of the string s
.
Finally, the recursive function is initiated with dfs(0)
to solve the entire string and the answer is checked against inf
to return either the minimum cuts or -1
if the partitioning isn't possible.
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 consider a small example to illustrate the solution approach step by step. Suppose we have the binary string s = "101101101"
which we want to partition into beautiful substrings following the given conditions.
-
Preprocess powers of 5 in binary: We first create a set
ss
of all binary strings that represent powers of 5 and are less than or equal to the length ofs
. Here is a small set for illustration:ss = {"1", "101", "10001", ...}
. -
Start with the recursive
dfs
function at index 0: We calldfs(0)
and begin to explore all substrings starting from index 0. -
Exploring substrings:
- We consider the first substring which is "1". Since "1" is in the
ss
set, it represents a power of 5. We then calldfs(1)
(which is the next index) to explore further the rest of the string "01101101". - We can't start a substring with '0', so
dfs(1)
immediately moves on todfs(2)
. - At index 2, we start with substring "1" again which is valid, and thus proceed to
dfs(3)
. - And so on, until
dfs(6)
considers the substring "101" which is also a power of 5.
- We consider the first substring which is "1". Since "1" is in the
-
Using Memoization: Assume
dfs(7)
is called, and the results for this index are calculated and stored. Ifdfs(7)
is called again during the recursive calls, we don't recompute it but rather retrieve the stored result. -
Recursive and Memoization Results: Continues until the entire string is processed. During this process:
- "101101101" at
dfs(0)
would proceed with "1" and calldfs(1)
. - "01101101" at
dfs(1)
skips the zero and then proceeds with "1" and calldfs(2)
. - "1101101" at
dfs(2)
proceeds with "1" and calldfs(3)
. - "101101" at
dfs(3)
then finds "101", callsdfs(6)
. - "101" at
dfs(6)
is a power of 5 itself, so it callsdfs(9)
, now at the base case because the end of the string is reached and it returns 0. - The result of
dfs(9)
is added to those ofdfs(6)
,dfs(3)
,dfs(2)
, anddfs(0)
to find the minimum number of beautiful substrings. For this example,dfs(9)
returns0
,dfs(6)
returns1
,dfs(3)
returns2
,dfs(2)
returns3
, anddfs(0)
finally returns4
.
- "101101101" at
-
End of Recursion: After applying the memoization and the entire recursion, if the returned value is infinity (
inf
), our function outputs-1
, indicating that it's not possible to partitions
according to the rules. However, fors = "101101101"
, the minimum number of beautiful substrings returned is 4 which are "1", "1", "1", and "101".
Therefore, for this example, our algorithm would successfully partition the string into the substrings {"1", "1", "1", "101â}, each of which is a binary representation of a power of 5, and since we've used 4 substrings, the output would be 4. If a partition was not found, our algorithm would return -1.
Solution Implementation
1from functools import lru_cache # Import the lru_cache decorator for memoization
2
3class Solution:
4 def minimumBeautifulSubstrings(self, binary_string: str) -> int:
5 # A decorator that caches the return values of the function it decorates
6 @lru_cache(maxsize=None)
7 def min_substrings_from_index(index: int) -> int:
8 # If we have reached the end of the binary_string, no more substrings needed
9 if index >= length:
10 return 0
11 # We cannot start with a '0' for a beautiful binary_string
12 if binary_string[index] == "0":
13 return float('inf') # Represents an impossible scenario
14
15 current_value = 0
16 best_result = float('inf')
17 # Iterate over the binary_string starting from current index
18 for j in range(index, length):
19 # Shift current_value by one bit and add the new bit
20 current_value = (current_value << 1) | int(binary_string[j])
21 # Check if current_value is a power of 5
22 if current_value in powers_of_five:
23 # Calculate the minimum substrings if we take current substring
24 # And then add 1 for the current one
25 best_result = min(best_result, 1 + min_substrings_from_index(j + 1))
26 return best_result
27
28 length = len(binary_string) # Total length of the binary_string
29 power_of_five = 1
30 # A set to store the powers of 5 values
31 powers_of_five = {power_of_five}
32 # Generate powers of 5 up to the length of the binary_string
33 for i in range(length):
34 power_of_five *= 5
35 powers_of_five.add(power_of_five)
36
37 # Start the search from index 0
38 result = min_substrings_from_index(0)
39 # Return -1 if there's no valid partition, otherwise the minimum substrings
40 return -1 if result == float('inf') else result
41
1import java.util.HashSet;
2import java.util.Set;
3
4public class Solution {
5 private Integer[] memoization; // memoization array to store results of subproblems
6 private String inputString; // the input string
7 private Set<Long> powersOfFive; // set containing powers of five
8 private int inputLength; // the length of the input string
9
10 // Method to calculate the minimum number of beautiful substrings
11 public int minimumBeautifulSubstrings(String s) {
12 inputLength = s.length();
13 this.inputString = s;
14 memoization = new Integer[inputLength];
15 powersOfFive = new HashSet<>();
16
17 // Precompute powers of 5 and add to the set
18 long power = 1;
19 for (int i = 0; i <= inputLength; ++i) {
20 powersOfFive.add(power);
21 power *= 5;
22 }
23
24 // Attempt to find the minimum beautiful substrings starting from the first character
25 int result = findMinimum(0);
26
27 // If the result exceeds the length of the input string, return -1, indicating no such decomposition
28 return result > inputLength ? -1 : result;
29 }
30
31 // Helper method to recursively find the minimum number of beautiful substrings starting at index i
32 private int findMinimum(int i) {
33 if (i >= inputLength) { // base case: if we've reached the end of the string
34 return 0;
35 }
36 if (inputString.charAt(i) == '0') { // no substring starting with a '0' can be beautiful
37 return inputLength + 1;
38 }
39 if (memoization[i] != null) { // return the precomputed value if available
40 return memoization[i];
41 }
42
43 long binaryValue = 0; // to store the numerical value of the substring in binary
44 int ans = inputLength + 1; // initialize the minimum with an upper bound
45
46 // Loop to consider all substrings starting at 'i'
47 for (int j = i; j < inputLength; ++j) {
48 binaryValue = (binaryValue << 1) | (inputString.charAt(j) - '0'); // accumulate the binary value
49 if (powersOfFive.contains(binaryValue)) { // if the binary value is a power of five
50 // Attempt to find the minimum starting at the next character and add 1 for the current substring
51 ans = Math.min(ans, 1 + findMinimum(j + 1));
52 }
53 }
54
55 // Store the result in the memoization array before returning
56 return memoization[i] = ans;
57 }
58}
59
1#include <unordered_set>
2#include <string>
3#include <cstring>
4#include <functional>
5using namespace std;
6
7class Solution {
8public:
9 // Function to calculate the minimum number of beautiful substrings.
10 int minimumBeautifulSubstrings(string s) {
11 unordered_set<long long> beautifulNumbers;
12 int n = s.size();
13 long long powerOfFive = 1;
14 // Populate a set with powers of 5. These represent the "beautiful numbers" in binary.
15 for (int i = 0; i <= n; ++i) {
16 beautifulNumbers.insert(powerOfFive);
17 powerOfFive *= 5;
18 }
19
20 // Array to store minimum beautiful substrings starting at each index
21 int minSubstrings[n];
22 memset(minSubstrings, -1, sizeof(minSubstrings));
23
24 // Lambda function to calculate minimum beautiful substrings using DFS
25 function<int(int)> dfs = [&](int idx) {
26 if (idx >= n) { // If the entire string has been processed
27 return 0; // Base case: no more substrings, so return 0
28 }
29 if (s[idx] == '0') { // Beautiful substrings cannot start with '0'
30 return n + 1; // Return a big number which will not be minimum
31 }
32 if (minSubstrings[idx] != -1) { // Check if already computed
33 return minSubstrings[idx];
34 }
35 long long num = 0;
36 int ans = n + 1; // Initialize the answer with a large number
37 for (int j = idx; j < n; ++j) {
38 num = (num << 1) | (s[j] - '0'); // Convert binary to decimal
39 if (beautifulNumbers.count(num)) { // If it's a beautiful number
40 // Take minimum of current answer and 1 plus the answer from next index
41 ans = min(ans, 1 + dfs(j + 1));
42 }
43 }
44 return minSubstrings[idx] = ans; // Memoize and return the answer
45 };
46
47 int ans = dfs(0); // Start DFS from index 0
48 return ans > n ? -1 : ans; // If answer is greater than n, no beautiful substring is found, return -1
49 }
50};
51
1function minimumBeautifulSubstrings(s: string): number {
2 // Set to hold powers of 5
3 const powersOfFive: Set<number> = new Set();
4 const lengthOfS = s.length;
5
6 // Array to hold the minimum beautiful substrings from each index
7 const minSubstrFromIndex: number[] = new Array(lengthOfS).fill(-1);
8
9 // Pre-calculate powers of 5 and store in the set
10 for (let i = 0, power = 1; i <= lengthOfS; ++i) {
11 powersOfFive.add(power);
12 power *= 5;
13 }
14
15 // Helper function to perform a depth-first search
16 const depthFirstSearch = (startIndex: number): number => {
17 // Base case: If we've reached the end of the string, return 0
18 if (startIndex === lengthOfS) {
19 return 0;
20 }
21 // If the current character is a '0', it cannot be beautiful, return large number
22 if (s[startIndex] === '0') {
23 return lengthOfS + 1;
24 }
25 // Use memoization to avoid recalculating
26 if (minSubstrFromIndex[startIndex] !== -1) {
27 return minSubstrFromIndex[startIndex];
28 }
29 // Initialize a large number for comparison
30 minSubstrFromIndex[startIndex] = lengthOfS + 1;
31
32 // Look ahead in the string to find valid beautiful substrings
33 for (let endIndex = startIndex, binaryValue = 0; endIndex < lengthOfS; ++endIndex) {
34 // Incrementally construct the binary value represented by the substring
35 binaryValue = (binaryValue << 1) | (s[endIndex] === '1' ? 1 : 0);
36 // Check if the current binary value is a power of 5
37 if (powersOfFive.has(binaryValue)) {
38 // If it is, update the minimum count and recurse
39 minSubstrFromIndex[startIndex] = Math.min(minSubstrFromIndex[startIndex], 1 + depthFirstSearch(endIndex + 1));
40 }
41 }
42
43 // Return the minimum count of beautiful substrings starting from the current index
44 return minSubstrFromIndex[startIndex];
45 };
46
47 // Start the depth-first search from the first character
48 const answer = depthFirstSearch(0);
49
50 // If the answer is larger than the length of the string, no valid solution exists, return -1
51 return answer > lengthOfS ? -1 : answer;
52}
53
Time and Space Complexity
The given Python code defines a recursive function dfs
to compute the minimum number of beautiful substrings. The time and space complexity analysis are as follows:
Time Complexity
The time complexity of this code is O(n^2)
. Here's the detailed reasoning:
- The function
dfs
is called recursively and it iterates over the entire length of the strings
for each starting indexi
. In the worst-case scenario, the inner loop could run through the remaining part of the string, which gives usn
iterations when starting at the first character,n-1
on the second, down to1
iteration at the last character. Summing these up gives us the arithmetic seriesn + (n-1) + ... + 1
, which has a sum of(n(n+1))/2
, resulting inO(n^2)
time complexity. - Each recursive call involves a constant time check and a loop which shifts and adds digits to
x
, but these operations areO(1)
for each individual call. - The calls to
min
and to checkif x in ss
are alsoO(1)
since checking membership in a set is constant time on average.
Space Complexity
The space complexity of this code is O(n)
for the following reasons:
- The recursive function
dfs
could at most be calledn
times consecutively before reaching the base case (a call stack of depthn
), thus the space complexity due to recursion isO(n)
. - The hash set
ss
which contains at mostn
elements also requiresO(n)
space. - There's a cache used to store the results of subproblems in
dfs
due to the@cache
decorator. The number of distinct subproblems is proportional to the length of the strings
, also leading to a space complexity ofO(n)
.
Therefore, the overall space complexity is dominated by these factors, resulting in O(n)
.
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
Backtracking Template Prereq DFS with States problems dfs_with_states Combinatorial search problems Combinatorial search problems involve finding groupings and assignments of objects that satisfy certain conditions Finding all permutations combinations subsets and solving Sudoku are classic combinatorial problems The time complexity of combinatorial problems often grows rapidly with the size of
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.