给定一个仅包含数字 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'] 的一个数字。来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/letter-combinations-of-a-phone-number
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
思路
1.我们不难想到本题的底层原理其实就是笛卡尔集,比如说任意三个字母,两两相乘,我们可以得到9种解法。
2.因此我们在这里使用的是回溯算法(backtrack) 【因为这里的遍历是不会涉及到不合理的答案,因此是不需要进行其他的一个判别,直接进行进行暴力枚举即可。】把每种结果都给它一一展示出来。
3.下面让我们结合代码来进行一个分析。
public List<String> letterCombinations(String digits) {
//准备一个集合List,用来存储String类型的一个数据。
List<String> conbinations = new ArrayList<>();
// 判空处理,如果为空直接返回!
if(digits.length() == 0) return conbinations;
//因为这个是参考leetcode解法,不是很理解这种方式的赋值,希望有懂的同学可以帮帮小白
// 声明一个map,通过键值对的方式进行电话号码的存储。(这里实际上是用到多态)父类引用指
// 向子类对象
Map<Character,String> phoneMap = new HashMap<Character,String>(){{
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(conbinations,digits,0,phoneMap,new StringBuilder());
//返回结果集合!
return conbinations;
}
//回溯算法的核心部分
public void backTrack(List<String> conbinations, String digits,int index,Map<Character,String> phoneMap, StringBuilder motivator){
//判断是否遍历完成本次存在的情况的最终情况
// 比如说(a可能会对应e,f,g),那么在这里我们主要判断是否已经到ae这个存在的情况的末尾了,就是e后面不会再有其他的结果了,然后它就把自己变成String类型,并存入到List中去。
if(index == digits.length()) conbinations.add(motivator.toString());
else{
//先取出每个String类型的单个字母
char goal = digits.charAt(index);
//对哈希表进行取值
String str = phoneMap.get(goal);
//循环遍历
for(int i = 0; i < str.length(); i++){
//把存在的每一种情况appen上去
motivator.append(str.charAt(i));
//回溯调用,index+1,实际上是往深度进行遍历(遍历到底),然后又重头开始
backTrack(conbinations,digits,index+1,phoneMap,motivator);
//简单来说就是这一层有两个方向可供选择,删除后,就会由下一层回退到上一层,走另
//外一条路,其他的就依次类推即可。
motivator.deleteCharAt(index);
}
}
}