057电话号码的字母组合

电话号码的字母组合

题目链接:https://leetcode.cn/problems/letter-combinations-of-a-phone-number/description/?envType=study-plan-v2\&envId=top-100-liked

我的解答:

复制代码
int[] first = new int[]{0, 3, 6, 9, 12, 15, 19, 22, 26};
StringBuilder sb = new StringBuilder();
public List<String> letterCombinations(String digits) {
    List<String> ans = new ArrayList<>();
    backtrack(digits, 0, ans);
    return ans;
}
public void backtrack(String digits, int cur, List<String> ans){
    if(cur == digits.length()){
        ans.add(sb.toString());
        return;
    }
    int curIndex = digits.charAt(cur) - '2';
    //每一种字母都选一遍
    for(int i=first[curIndex]; i<first[curIndex+1]; i++){
        char ch = (char) (i + 'a');
        sb.append(ch);
        backtrack(digits, cur+1, ans);
        //回溯
        sb.deleteCharAt(cur);
    }
}

分析:

​ 代码的时间复杂度为O(3^m * 4^n),空间复杂度为O(m+n),其中 m 是输入中对应 3 个字母的数字个数(包括数字 2、3、4、5、6、8),n 是输入中对应 4 个字母的数字个数(包括数字 7、9),m+n 是输入数字的总个数。

​ 解题思路:采用递归+回溯解题。用first数组维护每个数字能表示的第一个字母的下标(字母下标范围为0~25),递归遍历到digits中的每个数字时,将此数字能表示的字母都选择一遍,直到digits遍历完最后一个数字后收集答案。

看了官方题解后的解答:

复制代码
//方法:回溯
//时间复杂度:O(3^m * 4^n),其中 m 是输入中对应 3 个字母的数字个数(包括数字 2、3、4、5、6、8),n 是输入中对应 4 个字母的数字个数(包括数字 7、9),m+n 是输入数字的总个数。
//空间复杂度:O(m+n)
public List<String> letterCombinations(String digits) {
    List<String> ans = new ArrayList<>();
    Map<Character, String> phoneMap = new HashMap<>(){{
        put('2', "abc");
        put('3', "def");
        put('4', "ghi");
        put('5', "jkl");
        put('6', "mno");
        put('7', "pqrs");
        put('8', "tuv");
        put('9', "wxyz");
    }};
    backtrack(digits, 0, new StringBuffer(), phoneMap, ans);
    return ans;
}
public void backtrack(String digits, int cur, StringBuffer sb, Map<Character, String> phoneMap, List<String> ans){
    if(cur == digits.length()){
        ans.add(sb.toString());
        return;
    }
    char digit = digits.charAt(cur);
    String letters = phoneMap.get(digit);
    int lettersCount = letters.length();
    for(int i=0; i<lettersCount; i++){
        sb.append(letters.charAt(i));
        backtrack(digits, cur + 1, sb, phoneMap, ans);
        sb.deleteCharAt(cur);
    }
}

分析:官方的解题思路与我的解答一致,唯一的区别在于,我选择用数组first维护每个数字代表的字母,而官方采用哈希表维护。

总结

  • 本题只需掌握基本的递归+回溯方法,唯一需要思考的就是如何维护每个数字代表的字母。
相关推荐
Asize20 小时前
初识DFS 与 BFS:递归、队列与图遍历
算法
罗西的思考1 天前
机器人 / 强化学习】HIL-SERL:人类在环驱动的具身智能进化框架
人工智能·算法·机器学习
美团技术团队2 天前
LongCat 开源 VitaBench 2.0:长期动态智能体基准新标杆
人工智能·算法
To_OC2 天前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode
To_OC2 天前
LC 208 实现 Trie 前缀树:曾被名字劝退,写完发现是送分题
javascript·算法·leetcode
BadBadBad__AK2 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
_清歌3 天前
DSpark 深度解读:DeepSeek-V4 如何用「半自回归」把推理速度提升 85%
算法
统计实现局3 天前
SVD 的三步走:双对角化、Givens 收敛、排序
算法
躬行见万象3 天前
《VLA 系列》UniLab 强化训练 | G1 机器人 |复现
算法