Bootstrap

LeetCode100之电话号码的字母组合(17)--Java

1.问题描述

 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。

给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。

        示例1

输入:digits = "23"
输出:["ad","ae","af","bd","be","bf","cd","ce","cf"]

        示例2

输入:digits = ""
输出:[]

        示例3 

输入:digits = "2"
输出:["a","b","c"]

        提示

  • 0 <= digits.length <= 4
  • digits[i] 是范围 ['2', '9'] 的一个数字。

        难度等级

        中等

        题目链接

        电话号码的字母组合

2.解题思路

        这道题是要我们根据输入的电话号码,获取手机键盘上号码对应字母的所有组合。从题目给的图我们可以知道,一个电话号码可以对应多个字母,这里我们可以采用回溯的思路,每次取出一个字母进行组合,组合完成之后再回退到原来的序列,选择新的字母进行组合。

        首先,我们需要用一个数组,将每一个按键对应的字母存储起来,这里我使用了一个String数组来存储,每一个String存储一个按键的所有字母,因为题目的按键是从2开始的,而我们数组的索引是从0开始的,所以后面根据按键取出字母时,要在号码的基础上-2,才能找到对应的字母集,其次,我们还需要定义一个List集合来存储所有的字母组合。

        //按键对应的字母
        String[] keyboard = new String[]{"abc","def","ghi","jkl","mno","pqrs","tuv","wxyz"};
        //存储结果的List集合
        List<String> data = new ArrayList<String>();

        在正式开始寻找组合之前,我们可以先对电话号码进行校验,如果电话号码是空号码,那么就无需寻找组合了,直接返回即可。

        //如果电话号码没有空号码,直接返回空数据
        if(digits.length() == 0){
            return data;
        }

        接着,我们可以使用一个递归方法来将每一个按键对应的字母取出来进行组合。

        我们首先要确定一下递归结束的条件,如果我们已经取完电话号码的所有号码(或者说数),那么我们就可以将当前的组合存入存储结果的List集合中,然后方法结束。

        //已经挑完最后一个数字
        if(index == digits.length()){
            data.add(sb.toString());
            return;
        }

        确定了递归条件之后,我们就可以来确定我们需要传入的参数了,首先,我们要传入每一个数对应的字母集,也就是我们上面定义的String数组;还需要传入题目给的电话号码,有了电话号码之后,还要传入一个索引index,用来记录我们去到了电话号码中的第几个数;同时,还要传入存储结果的List集合来收集组合;最后,因为我们要不断的回溯,对取出的字母进行修改,如果使用String不断的对String进行修改,效率不高,所以我们可以传入一个StringBuilder或者StringBuffer来对已经取出的字母进行修改。

    public void backtrack(String[] keyboard,String digits,int index,List<String> data,StringBuilder sb)

        确定完递归参数之后,我们就可以来实现递归逻辑了。首先,我们要取出当前电话号码中要处理的数,我们可以通过传入的index索引直接定位到,将数取出来后找到它在键盘上对应的字母集,要注意的事,我们这里从电话号码中取出的数是一个ASCII码的字符,所以我们来减去0的ASCII码,才能得到真正的号码数,同时一开始就提到了,我们的数组是从0开始的,号码是从2开始的,所以我们还要从上面的基础上-2才能取出对应的字母集。

        //选择第index个数字对应的字母集
        String chs = keyboard[digits.charAt(index) - '0' - 2];

        接着,我们用一个for循环遍历字母集,每次取出一个字母放入当前组合中,然后递归调用当前这个获取所有字母组合的函数,将index+1来获取下一个索引对应的字母集加入到组合中,找到以当前字母为子组合的所有组合之后,还要将当前字母从子组合中移除,防止影响下一个字母的组合。

        //遍历字母集进行组合
        for(int i = 0;i < chs.length();i++){
            //取出一个字母添加到新组合中
            sb.append(chs.charAt(i));
            //递归获取所有组合
            backtrack(keyboard,digits,index+1,data,sb);
            //回溯
            sb.deleteCharAt(sb.length()-1);
        }

        找到所以的字母组合之后,将数据集合返回即可。

3.代码展示

class Solution {
    public List<String> letterCombinations(String digits) {
        //存储结果的List集合
        List<String> data = new ArrayList<String>();
        //如果电话号码没有空号码,直接返回空数据
        if(digits.length() == 0){
            return data;
        }
        //按键对应的字母
        String[] keyboard = new String[]{"abc","def","ghi","jkl","mno","pqrs","tuv","wxyz"};
        //调用递归回溯函数进行组合
        backtrack(keyboard,digits,0,data,new StringBuilder());
        //返回结果
        return data;
    }
    //递归所有的字母组合
    public void backtrack(String[] keyboard,String digits,int index,List<String> data,StringBuilder sb){
        //已经挑完最后一个数字
        if(index == digits.length()){
            data.add(sb.toString());
            return;
        }
        //选择第index个数字对应的字母集
        String chs = keyboard[digits.charAt(index) - '0' - 2];
        //遍历字母集进行组合
        for(int i = 0;i < chs.length();i++){
            //取出一个字母添加到新组合中
            sb.append(chs.charAt(i));
            //递归获取所有组合
            backtrack(keyboard,digits,index+1,data,sb);
            //回溯
            sb.deleteCharAt(sb.length()-1);
        }
    }

}

4.总结

        这道题是一个十分明显的递归回溯题目,我们只要将号码的数对应的字母挨个取出来,进行拼凑组合就可以搞定了。这道题就简单的啰嗦到这里,祝大家刷题愉快~

;