When I first looked at this problem, I immediately recognized it as a DFS / Backtracking problem. As I read more about solving backtracking problems, one key idea stood out:

“Think from the perspective of a node rather than the whole tree.”

So, first I converted the input to a proper vector to process.

private:
    vector<char> mapping(char digit){
        int INTDigit = digit - '0';
        vector<char> result;
        switch(INTDigit){
            case 2: return {'a', 'b', 'c'};
            case 3: return {'d', 'e', 'f'};
            case 4: return {'g', 'h', 'i'};
            case 5: return {'j', 'k', 'l'};
            case 6: return {'m', 'n', 'o'};
            case 7: return {'p', 'q', 'r', 's'};
            case 8: return {'t', 'u', 'v'};
            case 9: return {'w', 'x', 'y', 'z'};
            default: return {};
        }
    }
public:
    vector<string> letterCombinations(string digits) {
        vector<vector<char>> vec;
        for(int i=0; i<digits.size(); i++){
            vec.push_back(mapping(digits[i]));
        }
    }

Now, looking at the parameters which I would be needing:

  1. vector globalResult to store all the combinations
  2. string localResult which I will keep modifying and use to store the permutations
  3. int vectorpos = 0; // To process each node / vector
int n = digits.size();
int vectorpos = 0;
vector<string> globalResult;

Now, I would need to traverse through the first vector only and with each character I would do dfs for the next vector, so, I would pass in vectorpos + 1 to the dfs.

for(int i=0; i<vec[vectorpos].size(); i++){
    string localResult = string(1, vec[vectorpos][i]);
    dfs(vec, localResult, globalResult, vectorpos + 1, n);
}

Now, comes the main part of writing the dfs

  1. Base case:
void dfs(vector<vector<char>>& vec, string localResult, vector<string>& globalResult, int vectorpos, int n){
    if(vectorpos >= n){
        globalResult.push_back(localResult);
        return ;
    }
}

Now since the vectors are not exhausted, I can process this (vectorpos) vector. Therefore, I will proceed with processing and then perform DFS on this vector from the start.

for(int i = 0; i < vec[vectorpos].size(); i++){
    localResult.push_back(vec[vectorpos][i]);
    dfs(vec, localResult, globalResult, vectorpos + 1, n);
}

It doesn’t end here! Since a reference to localResult is passed, when we come back to this vector we would be appending the next character to this vector, along with the last inserted character.

Example:

["a", "b", "c"]

For i = 0 We would be inserting: localResult = “a” + dfs(….) For i = 1 It would become “a” + “b” + …

Therefore we need to remove “a” as well, so we do a pop_back();

Complete function:

for(int i=0; i < vec[vectorpos].size(); i++){
        localResult.push_back(vec[vectorpos][i]);
        dfs(vec, localResult, globalResult, vectorpos + 1, n);
        localResult.pop_back();
    }

So, complete code becomes:

class Solution {
private:
    vector<char> mapping(char digit){
        int INTDigit = digit - '0';
        vector<char> result;
        switch(INTDigit){
            case 2: return {'a', 'b', 'c'};
            case 3: return {'d', 'e', 'f'};
            case 4: return {'g', 'h', 'i'};
            case 5: return {'j', 'k', 'l'};
            case 6: return {'m', 'n', 'o'};
            case 7: return {'p', 'q', 'r', 's'};
            case 8: return {'t', 'u', 'v'};
            case 9: return {'w', 'x', 'y', 'z'};
            default: return {};
        }
    }

void dfs(vector<vector<char>>& vec, string localResult, vector<string>& globalResult, int vectorpos, int n){
    if(vectorpos >= n){
        globalResult.push_back(localResult);
        return ;
    }
    for(int i=0; i < vec[vectorpos].size(); i++){
        localResult.push_back(vec[vectorpos][i]);
        dfs(vec, localResult, globalResult, vectorpos + 1, n);
        localResult.pop_back();
    }
}

public:
    vector<string> letterCombinations(string digits) {
        if(digits.size() == 0) return {};
        vector<vector<char>> vec;
        for(int i=0; i<digits.size(); i++){
            vec.push_back(mapping(digits[i]));
        }
        int n = digits.size();
        int vectorpos = 0;
        vector<string> globalResult;
        for(int i=0; i<vec[vectorpos].size(); i++){
            string localResult = string(1, vec[vectorpos][i]);
            dfs(vec, localResult, globalResult, vectorpos + 1, n);
        }
        return globalResult;
    }
};