1639. Number of Ways to Form a Target String Given a Dictionary - Detailed Explanation

Free Coding Questions Catalog
Boost your coding skills with our essential coding questions catalog. Take a step towards a better tech career now!

Problem Statement

You are given a list of words, where each word has the same length m, and a string target. You need to determine how many ways you can form the target string by selecting one letter from each column of the words array. Each column in words represents a position from which you can pick a character (one from any word in that column). Return the number of ways modulo 10^9 + 7 (to handle large results).

Example 1:

Input:

words = ["acca", "bbbb", "caca"] target = "aba"

Output:

6

Explanation: There are 6 ways to form "aba" by picking letters from each column:

  • 'a' from "acca", 'b' from "bbbb", 'a' from "acca"
  • 'a' from "acca", 'b' from "bbbb", 'a' from "caca"
  • 'a' from "acca", 'b' from "bbbb", 'a' from "caca"
  • 'a' from "acca", 'b' from "bbbb", 'a' from "acca"
  • 'a' from "caca", 'b' from "bbbb", 'a' from "acca"
  • 'a' from "caca", 'b' from "bbbb", 'a' from "caca"

(Each bullet shows one possible selection of letters from the columns of the words list to form "aba".)

Constraints:

  • 1 <= words.length <= 1000
  • 1 <= words[i].length, target.length <= 100
  • words[i] consists of lowercase English letters.
  • target consists of lowercase English letters.

Hints to Solve the Problem

  1. Brute Force Approach: Consider generating all possible sequences of characters (by choosing one character from each column) and check which ones match the target. This is a direct approach but becomes inefficient as the size of input grows.
  2. Optimized Approach: Use dynamic programming (DP). Precompute the frequency of each character in each column of words. Then, use DP to count ways to form the target by iterating through target characters and columns.

Approach 1: Brute Force (Exhaustive Search)

Explanation:

This approach tries all possible ways of picking characters from the columns and counts the ones that form the target. We can use recursion or DFS to explore every combination:

  • Recursively go through each column index and try to match the current target character with letters from that column.
  • If a word in the current column has the required character for the current position of target, include that choice and move to the next character of target (and next column).
  • Also, consider the possibility of skipping the current column (not taking any character from this column for the target) and move to the next column.
  • Count all the combinations that successfully build the entire target string.

This brute force approach is straightforward but inefficient for large inputs because it explores every combination of choices (exponential time).

Python Code (Brute Force)

Python3
Python3

. . . .

Complexity Analysis (Brute Force)

  • Time Complexity: Exponential, approximately O(2^m) in the worst case (where m is the number of columns), because for each column, we decide to use it or skip it.
  • Space Complexity: O(m + n) due to the recursion stack.

Java Code (Brute Force):

Java
Java

. . . .

Approach 2: Optimized Dynamic Programming (Frequency Count per Column)

Explanation:

To overcome the inefficiency of brute force, we use dynamic programming with precomputed character frequencies:

  • Precompute Frequencies: First, count how many of each letter appears in each column of words. For example, in column 0 of words, count how many 'a', 'b', 'c', etc. appear among all the words. Store these counts in a 2D array freq[column_index][char].

  • DP State Definition: Use a DP table dp[i][j] where:

    • i represents how many characters of the target have been formed so far (considering target prefix of length i).
    • j represents how many columns of words have been considered so far.
    • dp[i][j] is the number of ways to form the first i characters of the target using the first j columns of words.
  • DP Transitions:

    • If we do not use the j-th column, then dp[i][j] can take all the ways from dp[i][j-1] (skip this column).

    • If we use the j-th column to contribute to the i-th character of target, then we look at the previous state dp[i-1][j-1] (formed first i-1 target chars with j-1 columns) and multiply it by the number of ways we can get the i-th target character from column j. The number of ways to get a specific character from column j is given by our precomputed freq.

    • Therefore:
      dp[i][j] = dp[i][j-1] + (dp[i-1][j-1] * freq[j-1][ target[i-1] ])
      (Here freq[j-1][ target[i-1] ] is the count of the character target[i-1] in column j-1 of words. We use j-1 and i-1 as indices because our DP table indices are 1-based for lengths.)

  • Initialization: dp[0][j] = 1 for all j, because there is exactly 1 way to form an empty target (do nothing, regardless of how many columns are considered).

  • We iterate through target length n and columns m to fill the DP table. The answer will be dp[n][m] — the number of ways to form all n characters of target using all m columns.

  • We apply modulo 10^9+7 at each addition or multiplication step to keep numbers within bounds.

This DP approach efficiently counts combinations without brute-forcing all sequences.

Python Code (Optimized DP)

Python3
Python3

. . . .

Complexity Analysis (Optimized DP)

  • Time Complexity: O(n * m) for filling the DP table plus O(m * 26) for building the frequency table.
  • Space Complexity: O(n * m) for the DP table and O(m * 26) for the frequency table.

Java Code (DP Solution):

Java
Java

. . . .

Edge Cases

  1. Minimum Input:
    • When words has one word of length 1 and target is a single character.
    • Example: words = ["a"], target = "a" → Output: 1.
  2. No Possible Formation:
    • If none of the columns contain the needed character at a particular target position, the output is 0.
  3. All Columns Identical:
    • When every column is the same, the number of ways depends on the frequency of the required character in each column.

Common Mistakes

  • Brute Force Pitfall:
    Not pruning search branches early can lead to a timeout for larger inputs.
  • DP Indexing:
    Off-by-one errors are common when translating between 0-indexed Python lists and conceptual 1-indexed DP tables.
  • Modulo Operations:
    Failing to apply the modulo (10^9+7) at every addition or multiplication step, which can result in integer overflow.
  • Word Break: Splitting a string into valid words using a dictionary.
  • Count Ways to Decode: Counting ways to decode a numeric string to alphabets (DP-based counting).
  • Subsequence Counting Problems: Such as counting distinct subsequences in a string.
TAGS
leetcode
CONTRIBUTOR
Design Gurus Team
-

GET YOUR FREE

Coding Questions Catalog

Design Gurus Newsletter - Latest from our Blog
Boost your coding skills with our essential coding questions catalog.
Take a step towards a better tech career now!
Explore Answers
How to answer what is cloud computing in an interview?
Is Okta a good company?
What is Datadog interview process?
Related Courses
Image
Grokking the Coding Interview: Patterns for Coding Questions
Grokking the Coding Interview Patterns in Java, Python, JS, C++, C#, and Go. The most comprehensive course with 476 Lessons.
Image
Grokking Modern AI Fundamentals
Master the fundamentals of AI today to lead the tech revolution of tomorrow.
Image
Grokking Data Structures & Algorithms for Coding Interviews
Unlock Coding Interview Success: Dive Deep into Data Structures and Algorithms.
Image
One-Stop Portal For Tech Interviews.
Copyright © 2025 Design Gurus, LLC. All rights reserved.
;