Solution: Palindromic Substrings

Let's solve the Palindromic Substrings problem using the Dynamic Programming pattern.

Statement

Given a string, s, return the number of palindromic substrings contained in it. A substring is a contiguous sequence of characters in a string. A palindrome is a phrase, word, or sequence that reads the same forward and backward.

Constraints:

  • 11 \leq s.length 1000\leq 1000

  • s consists of only lowercase English characters.

Solution

So far, you’ve probably brainstormed some approaches and have an idea of how to solve this problem. Let’s explore some of these approaches and figure out which to follow based on considerations such as time complexity and implementation constraints.

Naive approach

A naive approach to this problem is to find all possible substrings and count the number of palindromic substrings. For example, consider the string “deed”. The number of substrings contained in “deed” is 10: “d”, “e”, “e”, “d”, “de”, “ee”, “ed”, “dee”, “eed”, and “deed”. Out of these 10 substrings, six are palindromes: “d”, “e”, “e”, “d”, “ee”, and “deed”. Therefore, the number of palindromic substrings in “deed” is six.

We get the required result, but at what cost? Since we’re checking every possible substring, the total number of substrings in a string of length nn is O(n2)O(n^2). The time required to check whether a string is a palindrome is O(n)O(n). Therefore, the time complexity of this algorithm is O(n3)O(n^3). Since we’re not using any extra space, the space complexity of this algorithm is O(1)O(1).

Optimized approach using dynamic programming

If we look at the example above, we notice that any substring of length 33 contains a substring of length 11 at the center. Although we had already checked all substrings of length 11, our algorithm rechecked them for substrings of longer lengths. This rechecking consumes a lot of time, which can be avoided by storing and reusing the results of the earlier computations. To do this, we can create a lookup table, dp, of size n×nn \times n, where nn is the length of the input string. Each cell dp[i][j] will store whether the string s[i..j] is a palindromic substring. If the cell dp[i][j] holds the result of the earlier computation, we will utilize it in O(1)O(1) lookup time instead of making a recursive call again.

  1. First, we initialize a count variable with 00, which will count the number of palindromic substrings in s.

  2. A lookup table is initialized with FALSE.

  3. Base case 1: The diagonal in the lookup table is populated with TRUE because any cell in the diagonal corresponds to a substring of length one, and any string of length one is always a palindrome. The number of one-letter palindromic strings (i.e., the number of elements in the diagonal) is also added to the count.

  4. Base case 2: We check whether all two-letter substrings are palindromes and update the count and dp accordingly. We do this by iterating over the string, comparing s[i] and s[i+1], and storing the result at dp[i][i+1]. After that, we also increment the count if the value of dp[i][i+1] is TRUE, which tells us that the two-letter substring was a palindrome.

  5. After these base cases, we check all substrings of lengths greater than two. However, we only compare the first and the last characters. The rest of the string is checked using the lookup table. For example, for a given string “zing”, we want to check whether “zin” is a palindrome. We’ll only compare ‘z’ and ‘n’ and check the value of dp[1][1], which will tell whether the remaining string “i”, represented by s[1..1], is a palindrome. We’ll take the logical AND of these two results and store it at dp[0][2] because “zin” is represented by the substring s[0..2]. This way, we’ll avoid redundant computations and check all possible substrings using the lookup table.

Let’s look at the following illustration to get a better understanding of the solution:

Let’s implement the algorithm as discussed above:

import java.util.stream.*;
import java.util.*;
class Main {
public static int countPalindromicSubstrings(String s) {
int count = 0;
boolean[][] dp = new boolean[s.length()][s.length()];
for (int i = 0; i < s.length(); i++) {
dp[i][i] = true;
count++;
}
for (int i = 0; i < s.length() - 1; i++) {
dp[i][i + 1] = (s.charAt(i) == s.charAt(i + 1));
count += dp[i][i + 1] ? 1 : 0;
}
for (int length = 3; length <= s.length(); length++) {
for (int i = 0, j = length - 1; j < s.length(); i++, j++) {
dp[i][j] = dp[i + 1][j - 1] && (s.charAt(i) == s.charAt(j));
count += dp[i][j] ? 1 : 0;
}
}
return count;
}
// Driver code
public static void main(String[] args) {
List<String> strings = new ArrayList<>(Arrays.asList("cat", "lever", "xyxxyz", "wwwwwwwwww", "tattarrattat"));
for (int i = 0; i < strings.size(); i++) {
System.out.println((i + 1) + ".\t Input string: '" + strings.get(i) + "'");
int result = countPalindromicSubstrings(strings.get(i));
System.out.println("\t Number of palindromic substrings: " + result);
Stream.generate(() -> "-").limit(100).forEach(System.out::print);
System.out.println();
}
}
}
Palindromic Substrings

Solution summary

To recap, the solution to this problem can be divided into the following five main parts:

  1. We count all one-letter substrings since any one-letter string is always a palindrome. These results are also stored in a lookup table to be used later.

  2. Next, the algorithm checks all two-letter substrings and updates the count and the lookup table accordingly.

  3. After these base cases, the algorithm checks all possible substrings of lengths greater than three. However, it only compares the first and last characters, and the rest of the substring is checked using the lookup table.

  4. Whenever a palindromic substring is found, the count and the lookup table are updated accordingly.

  5. After checking all possible substrings, the algorithm terminates and returns the count of the palindromic substrings.

Time complexity

The time complexity of this algorithm is O(n2)O(n^2), where nn is the length of the input string. This is the time required to populate the lookup table.

Space complexity

The space complexity of this algorithm is O(n2)O(n^2), where nn is the length of the input string. This is the space occupied by the lookup table.

Level up your interview prep. Join Educative to access 70+ hands-on prep courses.