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维护每个数字代表的字母,而官方采用哈希表维护。

总结

  • 本题只需掌握基本的递归+回溯方法,唯一需要思考的就是如何维护每个数字代表的字母。
相关推荐
玖釉-2 小时前
下一个排列:从字典序到原地算法的完整推导
数据结构·c++·windows·算法
IronMurphy2 小时前
【算法五十】62. 不同路径
算法
影寂ldy3 小时前
C#一维数组
算法
过期动态3 小时前
【LeetCode 热题 100】移动零
java·数据结构·算法·leetcode·职场和发展·rabbitmq
计算机安禾4 小时前
【算法分析与设计】第10篇:下界理论与NP完全性初步
大数据·人工智能·算法
水木流年追梦5 小时前
大模型入门-大模型分布式训练2
开发语言·分布式·python·算法·正则表达式·prompt
sali-tec5 小时前
C# 基于OpenCv的视觉工作流-章78-KRT测量
图像处理·人工智能·数码相机·opencv·算法·计算机视觉
菜菜的顾清寒5 小时前
力扣HOT100(32)二叉树的中序遍历
数据结构·算法·leetcode
x2c5 小时前
数据结构:线性表中链表的建立和基本操作(C)
算法