电话号码的字母组合
力扣原题链接
问题描述
给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。
给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。
示例
示例 1:
输入:digits = “23”
 输出:[“ad”,“ae”,“af”,“bd”,“be”,“bf”,“cd”,“ce”,“cf”]
示例 2:
输入:digits = “”
 输出:[]
示例 3:
输入:digits = “2”
 输出:[“a”,“b”,“c”]
解题思路
这是一个典型的回溯算法问题。我们需要根据数字到字母的映射,将给定的数字字符串转换为所有可能的字母组合。
代码思路
- 构建数字到字母的映射表: 首先构建一个数组 LETTERS,用于存储数字到字母的映射,数组下标代表数字,数组元素代表对应的字母字符串。
- 初始化结果列表: 创建一个空的列表 combinations,用于存储最终的字母组合结果。
- 回溯搜索: 定义一个回溯函数 backtrack,其参数包括当前数字字符串digits、当前处理的索引index、当前的字母组合路径path。
- 结束条件: 如果当前路径长度等于数字字符串的长度,则将当前路径加入结果列表,并返回。
- 选择列表: 获取当前数字对应的字母集合。
- 遍历选择: 遍历当前数字对应的字母集合,对每个字母进行递归搜索。
- 做出选择: 将当前字母加入路径。
- 递归进入下一层: 递归调用回溯函数,传入新的索引 index+1,继续搜索下一个数字对应的字母。
- 撤销选择: 回溯到上一层时,将当前选择的字母从路径中删除,继续遍历下一个字母。

Java解题
垃圾版
class Solution {
    Map<Integer, String> en = new HashMap<>(); 
    List<String> res = new ArrayList<>();
    public List<String> letterCombinations(String digits) {
        if(digits.equals("")|| digits == null) return res;
        en.put(0,"");
        en.put(1,"");
        en.put(2,"abc");
        en.put(3,"def");
        en.put(4,"ghi");
        en.put(5,"jkl");
        en.put(6,"mno");
        en.put(7,"pqrs");
        en.put(8,"tuv");
        en.put(9,"wxyz");
        String s = new String();
        backtract(digits,0,s);
        return res;
    }
    public void backtract(String digits,int index,String s){
        if(index == digits.length()){
            res.add(s);
            return;
        }
        String arr = en.get(Integer.parseInt(String.valueOf(digits.charAt(index))));
        for(int i =0 ;i<arr.length();i++){
            s += arr.charAt(i);
            backtract(digits,index+1,s);
            s = s.substring(0,s.length()-1);
        }
    }
}
优化版
- 直接使用String 数组就行,数组下标和元素就是一个二维映射,不用使用hash表的计算查询。
- 收集结果使用StringBuilder,避免String进行拼接,裁剪时的新建。
import java.util.*;
class Solution {
    // 存储数字到字母的映射关系
    private static final String[] LETTERS = {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};
    // 存储最终结果的列表
    private List<String> combinations = new ArrayList<>();
    public List<String> letterCombinations(String digits) {
        if (digits.length() == 0) {
            return combinations;
        }
        backtrack(digits, 0, new StringBuilder());
        return combinations;
    }
    // 回溯函数
    private void backtrack(String digits, int index, StringBuilder path) {
        // 如果路径长度等于数字字符串的长度,将路径加入结果列表
        if (index == digits.length()) {
            combinations.add(path.toString());
            return;
        }
        // 获取当前数字对应的字母集合
        String letters = LETTERS[digits.charAt(index) - '0'];
        // 遍历当前数字对应的字母集合
        for (char letter : letters.toCharArray()) {
            // 做出选择
            path.append(letter);
            // 递归进入下一层决策树
            backtrack(digits, index + 1, path);
            // 撤销选择
            path.deleteCharAt(path.length() - 1);
        }
    }
}
通过回溯算法,我们可以找出给定数字字符串能表示的所有字母组合。


![[项目实践]---RSTP生成树](https://img-blog.csdnimg.cn/direct/506c838a49524a5a946553ef54a33fdd.png)
















