131. Palindrome Partitioning
Problem Description
The problem asks for all ways a given string s
can be split such that each substring in the partition is a palindrome. A palindrome is a string that reads the same forwards and backwards, such as 'racecar'.
For example, if the input string is "aab"
, then there are two ways to partition it into substrings that are all palindromes: ["aa", "b"]
and ["a", "a", "b"]
.
Intuition
The core idea behind the solution is to use depth-first search (DFS). We want to explore all possible partitions and whenever we encounter a partition where all substrings are palindromes, we add that partition to our answer.
Here, DFS is used to recursively build partitions. Starting from the beginning of the string, we check every possible end index for a substring that could be a palindrome. If a substring from the current start index to the current end index is a palindrome, then we add it to the current partition list and invoke DFS from the next index. If it's not, we skip the current index and move to the next.
To optimize the identification of palindromes, we use dynamic programming. A 2D table f
is maintained where f[i][j]
is True
if the substring from index i
to j
is a palindrome. The elements in this table are filled in by checking if the two ends of the substring are equal and if the inside substring (from i+1
to j-1
) is also a palindrome.
The recursive DFS approach combined with dynamic programming allows us to efficiently compute all the ways to partition the string into substrings that are palindromes.
Learn more about Dynamic Programming and Backtracking patterns.
Solution Approach
The solution uses a DFS algorithm combined with dynamic programming to efficiently find all possible palindrome partitioning of the string s
. Let's walk through the key parts of the implementation.
Firstly, the dynamic programming table f
is prepared, which is a square matrix, where f[i][j]
indicates whether the substring from i
to j
(inclusive) is a palindrome. This precomputation avoids redundant checks and optimizes the process of verifying palindromes during the DFS traversal.
Here's how the table f
is populated:
- Initialize
f
as ann x n
matrix filled withTrue
, as every single character is a palindrome by itself. - The table is then filled in a bottom-up manner. For substrings longer than one character (from
n-1
to0
fori
and fromi+1
ton
forj
), we check if the two ends are the same,s[i] == s[j]
, and if the inside substringf[i+1][j-1]
is a palindrome.
Once the table is prepared, we use the DFS approach to build the partitions:
- A recursive function
dfs
takes an indexi
as a parameter, indicating the current starting point of the substring being considered. - The recursive function works as follows:
- If
i
equals the length of the stringn
, it means we reached the end of the string and thus, have a valid partition we can add to the answer listans
. - For other cases, we check all possible end indices
j
(fromi
ton-1
) and iff[i][j]
isTrue
, indicating a palindrome, we append this substring to the current partition listt
. - We then recursively call
dfs(j + 1)
, which will attempt to find palindrome partitions for the remaining substring. - Afterwards, we must backtrack by removing the last substring added to
t
before moving on to the next index.
- If
The DFS is initiated by calling dfs(0)
, meaning it starts with an empty partition and it looks at the string from the very start. The result is accumulated in the list ans
, which eventually contains all the possible palindrome partitionings.
Note that ans
is a list of lists, where each sublist represents a distinct palindrome partitioning of s
. The variable t
represents the current state of the partition list under consideration at each step of the DFS.
By using a combination of DFS for enumeration, dynamic programming for palindrome checking, and backtracking for generating partitions, the solution efficiently enumerates all palindrome partitions of the string s
.
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 an example to illustrate the solution approach using the string "aab"
.
Step 1: First, we prepare the dynamic programming table f
. The length of the string s
is 3
, so our table will be a 3x3 matrix. Initially, f[i][i]
(all positions where i == j
) are set to True
since all characters are palindromes by themselves.
1f = [ 2 [True, False, False], 3 [False, True, False], 4 [False, False, True] 5]
Step 2: Now we fill the remaining f[i][j]
values where i != j
. This involves checking substrings of length 2 and above.
- For
i = 1
andj = 2
("aa"
),s[i]
is equal tos[j]
and the inside substring (which is empty) is "trivially" a palindrome. So,f[1][2]
is set toTrue
. - For
i = 0
andj = 1
("ab"
),s[i]
is not equal tos[j]
, sof[0][1]
remainsFalse
. - For
i = 0
andj = 2
("aab"
),s[i]
is not equal tos[j]
, sof[0][2]
remainsFalse
.
Our DP table now looks like this:
1f = [ 2 [True, False, False], 3 [False, True, True], // "aa" is a palindrome 4 [False, False, True] 5]
Step 3: We start a DFS traversal to build palindrome partitions.
- We call
dfs(0)
and look for all palindromic substrings starting at index0
. - For
i = 0
, we consider substring"a"
(f[0][0]
isTrue
), and so we add["a"]
to our current partitiont
and calldfs(1)
. - Inside
dfs(1)
, we again find a palindromic substring"a"
(f[1][1]
isTrue
), so we add["a"]
tot
and calldfs(2)
. - Finally, in
dfs(2)
, we add"b"
tot
, sincef[2][2]
isTrue
, and reach the end of the string. We now have["a", "a", "b"]
and add this partition to our answer listans
. - We backtrack to
dfs(1)
and continue the loop forj = 2
, now"aa"
is a palindrome asf[1][2]
isTrue
, so we add["aa"]
tot
and calldfs(3)
. - In
dfs(3)
, we've reached the end of the string again, so we now have["a", "aa"]
which is a valid partition that we add toans
.
The DFS finishes, and our answer ans
contains all possible palindrome partitioning for the input string "aab"
:
1ans = [ 2 ["a", "a", "b"], 3 ["aa", "b"] 4]
We successfully obtained all partitions where each substring is a palindrome using a combination of DFS, dynamic programming, and backtracking.
Solution Implementation
1class Solution:
2 def partition(self, s: str) -> List[List[str]]:
3 def dfs(start_index: int):
4 # Base case: if start_index reaches the end of the string,
5 # add a copy of the current partition to the answer list
6 if start_index == length:
7 result.append(current_partition[:]) # Make a shallow copy
8 return
9
10 # Iterate over the substring starting from start_index
11 for end_index in range(start_index, length):
12 # If the substring s[start_index:end_index+1] is a palindrome,
13 # we proceed to add it to the current partition and recurse
14 if palindrome_flags[start_index][end_index]:
15 current_partition.append(s[start_index:end_index + 1])
16 dfs(end_index + 1) # Recurse with the next starting index
17 current_partition.pop() # Backtrack
18
19 # Get the length of the input string
20 length = len(s)
21
22 # Initialize a 2D list to store palindrome flags
23 palindrome_flags = [[True] * length for _ in range(length)]
24
25 # Fill the palindrome_flags list in reverse order
26 for i in range(length - 1, -1, -1):
27 for j in range(i + 1, length):
28 # A substring is a palindrome if the first and last characters are the same,
29 # and if the substring between them is also a palindrome
30 palindrome_flags[i][j] = s[i] == s[j] and palindrome_flags[i + 1][j - 1]
31
32 # Initialize the result list and the list to store the current partition
33 result = []
34 current_partition = []
35
36 # Start the DFS traversal from index 0
37 dfs(0)
38
39 return result
40
1class Solution {
2 private int stringLength;
3 private String inputString;
4 private boolean[][] palindromeTable;
5 private List<String> currentPartition = new ArrayList<>();
6 private List<List<String>> allPartitions = new ArrayList<>();
7
8 public List<List<String>> partition(String s) {
9 stringLength = s.length();
10 inputString = s;
11 palindromeTable = new boolean[stringLength][stringLength];
12
13 // Initialize the palindrome table with true for all entries
14 for (int i = 0; i < stringLength; ++i) {
15 Arrays.fill(palindromeTable[i], true);
16 }
17
18 // Populate the palindrome table with actual palindrome information
19 for (int i = stringLength - 1; i >= 0; --i) {
20 for (int j = i + 1; j < stringLength; ++j) {
21 palindromeTable[i][j] = (s.charAt(i) == s.charAt(j)) && palindromeTable[i + 1][j - 1];
22 }
23 }
24
25 // Start the depth-first search from the beginning of the string
26 performDfs(0);
27 return allPartitions;
28 }
29
30 private void performDfs(int startIndex) {
31 // If the current start index reaches the end of the string, we've found a complete partition
32 if (startIndex == inputString.length()) {
33 allPartitions.add(new ArrayList<>(currentPartition));
34 return;
35 }
36
37 // Explore further partitions starting from the current index
38 for (int endIndex = startIndex; endIndex < stringLength; ++endIndex) {
39 // If the substring starting at startIndex and ending at endIndex is a palindrome
40 if (palindromeTable[startIndex][endIndex]) {
41 // Add the palindrome substring to the current partition
42 currentPartition.add(inputString.substring(startIndex, endIndex + 1));
43
44 // Continue searching for palindromes from the next index after the current palindrome
45 performDfs(endIndex + 1);
46
47 // Backtrack and remove the last added palindrome from the current partition
48 currentPartition.remove(currentPartition.size() - 1);
49 }
50 }
51 }
52}
53
1#include <vector>
2#include <string>
3#include <cstring>
4#include <functional>
5
6class Solution {
7public:
8 std::vector<std::vector<std::string>> partition(std::string s) {
9 int length = s.size();
10
11 // Create a table to record if the substring s[i..j] is a palindrome.
12 bool palindromeTable[length][length];
13 memset(palindromeTable, true, sizeof(palindromeTable));
14
15 // Fill the palindromeTable using dynamic programming.
16 for (int i = length - 1; i >= 0; --i) {
17 for (int j = i + 1; j < length; ++j) {
18 // A substring is a palindrome if its outer characters are equal
19 // and if its inner substring is also a palindrome.
20 palindromeTable[i][j] = (s[i] == s[j]) && palindromeTable[i + 1][j - 1];
21 }
22 }
23
24 // Create a vector to store all palindrome partitioning results.
25 std::vector<std::vector<std::string>> result;
26
27 // Temporary vector to store current partitioning.
28 std::vector<std::string> tempPartition;
29
30 // Recursively search for palindrome partitions starting from index 0.
31 std::function<void(int)> depthFirstSearch = [&](int start) {
32 // If we've reached the end of the string, add the current partitioning to results.
33 if (start == length) {
34 result.push_back(tempPartition);
35 return;
36 }
37
38 // Explore all possible partitionings.
39 for (int end = start; end < length; ++end) {
40 // If the substring starting from 'start' to 'end' is a palindrome
41 if (palindromeTable[start][end]) {
42 // Push the current palindrome substring to the temporary partitioning.
43 tempPartition.push_back(s.substr(start, end - start + 1));
44
45 // Move to the next part of the string.
46 depthFirstSearch(end + 1);
47
48 // Backtrack to explore other partitioning possibilities.
49 tempPartition.pop_back();
50 }
51 }
52 };
53
54 // Start the depth-first search from the first character.
55 depthFirstSearch(0);
56
57 // Return all the palindrome partitioning found.
58 return result;
59 }
60};
61
1function partition(s: string): string[][] {
2 const length = s.length;
3 // 'isValidPalindrome' table to keep track of valid palindrome substrings.
4 const isValidPalindrome: boolean[][] = new Array(length)
5 .fill(0)
6 .map(() => new Array(length).fill(true));
7
8 // Fill the table with the correct values, bottom-up manner.
9 for (let start = length - 1; start >= 0; --start) {
10 for (let end = start + 1; end < length; ++end) {
11 isValidPalindrome[start][end] = (s[start] === s[end]) && isValidPalindrome[start + 1][end - 1];
12 }
13 }
14
15 // This will hold all possible palindrome partitions.
16 const allPartitions: string[][] = [];
17 // 'currentPartition' temporarily stores one possible partition.
18 const currentPartition: string[] = [];
19
20 // Helper function to perform depth-first search for partitions.
21 const dfs = (index: number) => {
22 // If the entire string has been processed, save the current partition.
23 if (index === length) {
24 allPartitions.push(currentPartition.slice());
25 return;
26 }
27 // Explore further partitions.
28 for (let end = index; end < length; ++end) {
29 // Check if the substring is a valid palindrome
30 if (isValidPalindrome[index][end]) {
31 // Add the palindrome substring to the current partition.
32 currentPartition.push(s.slice(index, end + 1));
33 // Recursively check for further palindromes from the end of the current substring.
34 dfs(end + 1);
35 // Backtrack to explore other possible partitions by removing the last added palindrome.
36 currentPartition.pop();
37 }
38 }
39 };
40
41 // Start the depth-first search from the beginning of the string.
42 dfs(0);
43 // Return all possible palindrome partitions found.
44 return allPartitions;
45}
46
Time and Space Complexity
Time Complexity
The given code first prepares a 2D boolean array f
, which is used to determine if a substring s[i:j]
is a palindrome. This preparation takes O(n^2) time because it gradually shrinks the substring window from both sides and each cell in f
is filled based on the result of palindrome checks on previous substrings.
The main part of the algorithm uses depth-first search (DFS) to build all possible palindrome partitions of the string. Every recursive call of dfs
goes through the string to find palindromes starting from the current index i
up to the end of the string. In the worst case scenario, each character could be the start of a palindrome substring, leading to a situation where the number of recursive calls is proportional to the Catalan numbers because each step can involve a choice of different ending indices for the palindrome. The nth Catalan number is given by the formula (2*n)! / ((n+1)!*n!)
which is approximately O(4^n / (n^(3/2)))
. This determines the number of possible partitions in the worst case and thus the recursive calls made.
Given that the length of the input string is n
, the time complexity is O(n^2) for the initial palindrome check array preparation plus the time complexity of the DFS traversal, which is O(4^n / (n^(3/2))). Hence, the overall worst-case time complexity is O(n^2 + 4^n / (n^(3/2)))
.
Space Complexity
The space complexity is influenced by two factors: the space taken by the 2D array f
and the space used by the recursion call stack.
- The 2D array
f
has sizen^2
, hence it uses O(n^2) space. - The depth of the recursion stack is at most
n
(the size of the strings
), because that's the deepest it can go (each character can be a separate palindrome). Furthermore, at each level of recursion a string is potentially added to the current partitiont
. Thus, we need to account for the space taken by the list of current palindromest
, which in the worst case can store up ton
strings.
Thus, the space complexity of the recursive calls and the space to store the current partitions is O(n).
However, we also have an ans
list that stores all the possible palindrome partitions. In the worst case, there can be exponential (Catalan number) many partitions, and the list of partitions can grow very large, significantly larger than the stack depth.
Combining these factors, the space complexity is O(n^2)
for the 2D array and O(4^n / (n^(3/2)))
for the partitions, which gives us the overall space complexity of O(n^2 + 4^n / (n^(3/2)))
.
Learn more about how to find time and space complexity quickly using problem constraints.
Given a sorted array of integers and an integer called target, find the element that
equals to the target and return its index. Select the correct code that fills the
___
in the given code snippet.
1def binary_search(arr, target):
2 left, right = 0, len(arr) - 1
3 while left ___ right:
4 mid = (left + right) // 2
5 if arr[mid] == target:
6 return mid
7 if arr[mid] < target:
8 ___ = mid + 1
9 else:
10 ___ = mid - 1
11 return -1
12
1public static int binarySearch(int[] arr, int target) {
2 int left = 0;
3 int right = arr.length - 1;
4
5 while (left ___ right) {
6 int mid = left + (right - left) / 2;
7 if (arr[mid] == target) return mid;
8 if (arr[mid] < target) {
9 ___ = mid + 1;
10 } else {
11 ___ = mid - 1;
12 }
13 }
14 return -1;
15}
16
1function binarySearch(arr, target) {
2 let left = 0;
3 let right = arr.length - 1;
4
5 while (left ___ right) {
6 let mid = left + Math.trunc((right - left) / 2);
7 if (arr[mid] == target) return mid;
8 if (arr[mid] < target) {
9 ___ = mid + 1;
10 } else {
11 ___ = mid - 1;
12 }
13 }
14 return -1;
15}
16
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.