Problem Statement (LeetCode 17)
Given a string containing digits from 2-9
inclusive, return all possible letter combinations that the number could represent. Return the answer in any order.
A mapping of digits to letters (just like on the telephone buttons) is given below:
2 -> "abc"
3 -> "def"
4 -> "ghi"
5 -> "jkl"
6 -> "mno"
7 -> "pqrs"
8 -> "tuv"
9 -> "wxyz"
Example 1:
Input: digits = "23"
Output: ["ad", "ae", "af", "bd", "be", "bf", "cd", "ce", "cf"]
Example 2:
Input: digits = ""
Output: []
Example 3:
Input: digits = "2"
Output: ["a", "b", "c"]
Optimal Approach: Backtracking
Intuition
This problem can be approached effectively using backtracking. For each digit in the input string, we need to generate all possible combinations of the corresponding letters. This is a typical case of combinatorial generation.
Why Backtracking Works
- Combinatorial Explosion: Each digit maps to a set of letters, and we need to explore all possible combinations. Backtracking allows us to explore all possibilities and efficiently build the combinations.
- Depth-First Search (DFS): The algorithm works by considering each possible letter for the current digit, then recursively proceeding to the next digit.
- Pruning: The recursion will terminate when all digits have been processed, ensuring that we generate exactly the required combinations.
LeetCode-Compatible C++ Code
class Solution {
public:
vector<string> letterCombinations(string digits) {
if (digits.empty()) {
return {}; // Return empty if no digits are provided
}
// Mapping digits to corresponding characters
vector<string> mapping = {
"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"
};
vector<string> result;
string current;
backtrack(digits, 0, current, result, mapping);
return result;
}
private:
void backtrack(const string& digits, int index, string& current, vector<string>& result, const vector<string>& mapping) {
// If the current combination is complete, add it to the result
if (index == digits.size()) {
result.push_back(current);
return;
}
// Get the current digit
int digit = digits[index] - '0';
string letters = mapping[digit];
// Try each letter corresponding to the current digit
for (char letter : letters) {
current.push_back(letter); // Add the letter to the current combination
backtrack(digits, index + 1, current, result, mapping); // Recurse for the next digit
current.pop_back(); // Backtrack, remove the last letter
}
}
};
Step-by-Step Explanation of Code
-
Input Check:
If thedigits
string is empty, we return an empty list because no combinations are possible. -
Mapping Digits to Letters:
We use a vectormapping
where each index corresponds to a digit (2-9), and each entry holds the string of letters that the digit maps to. For example, index 2 holds the string"abc"
, and index 3 holds"def"
. -
Backtracking Function:
Thebacktrack
function recursively builds the letter combinations. It accepts the following parameters:digits
: The input string of digits.index
: The current position in the digits string we are processing.current
: The current combination of letters being formed.result
: The vector to store the valid combinations once they are completed.mapping
: The array of letter strings corresponding to each digit.
-
Base Case:
Whenindex
equals the size ofdigits
, we have successfully formed a combination of letters, so we add it toresult
. -
Recursive Case:
For each character corresponding to the current digit, we add it to thecurrent
string, recurse for the next digit, and then backtrack by removing the character and trying the next possibility. -
Return the Result:
Once the recursion finishes, theresult
vector contains all the combinations, which is then returned.
Dry Run with Example
Input: digits = "23"
Step 1: Start the backtracking with index = 0
, current = ""
, and an empty result.
-
At
index = 0
, the digit is '2' which maps to the letters "abc".- For 'a',
current = "a"
, recurse toindex = 1
. - For 'b',
current = "b"
, recurse toindex = 1
. - For 'c',
current = "c"
, recurse toindex = 1
.
- For 'a',
-
At
index = 1
, the digit is '3' which maps to the letters "def".- For 'd', add "ad", recurse to
index = 2
, add to result. - For 'e', add "ae", recurse to
index = 2
, add to result. - For 'f', add "af", recurse to
index = 2
, add to result.
- For 'd', add "ad", recurse to
Final Output: ["ad", "ae", "af", "bd", "be", "bf", "cd", "ce", "cf"]
Alternative Approaches & Why Not?
Approach | Time Complexity | Space Complexity | Why Not? |
---|---|---|---|
Backtracking (Current Approach) | O(4^n) | O(n) | This approach efficiently generates all combinations in a depth-first manner. |
Iterative Approach (Queue) | O(4^n) | O(n) | Can be used to iteratively build the combinations, but is more complicated compared to recursion. |
Memoization (DP) | O(4^n) | O(n) | Not necessary since we aren't repeating calculations that can be memoized. |
Why Backtracking is the Best
- Time Complexity: O(4^n), where
n
is the length of the input string. This is because there are up to 4 options for each digit (since the longest mapping has 4 letters: "pqrs"). - Space Complexity: O(n), where
n
is the length of the input string, for the recursion stack and thecurrent
string. - Clean and Efficient: Backtracking is a clean and direct way to generate combinations in a depth-first manner, ensuring all possible combinations are covered.
Complexity Analysis
-
Time Complexity: O(4^n)
There are up to 4 letters for each digit, so in the worst case, there are 4^n combinations. Each recursive call processes one combination. -
Space Complexity: O(n)
The recursion depth is at mostn
(the length of the input), and each recursive call holds a string of lengthn
in thecurrent
variable.
Conclusion
✅ Best Approach: Backtracking
✅ Time Complexity: O(4^n)
✅ Space Complexity: O(n)
✅ Optimal for Generating Combinations Efficiently
Recommended Problems for Practice
🚀 Master the Backtracking Technique to Generate All Possible Combinations!