【问题描述】[中等]

【解答思路】

用哈希表/数组存储每个数字对应的所有可能的字母,然后进行回溯操作。

回溯过程中维护一个字符串,表示已有的字母排列(如果未遍历完电话号码的所有数字,则已有的字母排列是不完整的)。该字符串初始为空。每次取电话号码的一位数字,从哈希表中获得该数字对应的所有可能的字母,并将其中的一个字母插入到已有的字母排列后面,然后继续处理电话号码的后一位数字,直到处理完电话号码中的所有数字,即得到一个完整的字母排列。然后进行回退操作,遍历其余的字母排列。

回溯算法用于寻找所有的可行解,如果发现一个解不可行,则会舍弃不可行的解。在这道题中,由于每个数字对应的每个字母都可能进入字母组合,因此不存在不可行的解,直接穷举所有的解即可。

1. 回溯 哈希表
class Solution {public List<String> letterCombinations(String digits) {List<String> combinations = new ArrayList<String>();if (digits.length() == 0) {return combinations;}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(combinations, phoneMap, digits, 0, new StringBuffer());return combinations;}public void backtrack(List<String> combinations, Map<Character, String> phoneMap, String digits, int index, StringBuffer combination) {if (index == digits.length()) {combinations.add(combination.toString());} else {char digit = digits.charAt(index);String letters = phoneMap.get(digit);int lettersCount = letters.length();for (int i = 0; i < lettersCount; i++) {combination.append(letters.charAt(i));backtrack(combinations, phoneMap, digits, index + 1, combination);combination.deleteCharAt(index);}}}
}
2. 回溯 数组存储
class Solution {// 数字到号码的映射private String[] map = {"abc","def","ghi","jkl","mno","pqrs","tuv","wxyz"};// 路径private StringBuilder sb = new StringBuilder();// 结果集private List<String> res = new ArrayList<>();public List<String> letterCombinations(String digits) {if(digits == null || digits.length() == 0) return res;backtrack(digits,0);return res;}// 回溯函数private void backtrack(String digits,int index) {if(sb.length() == digits.length()) {res.add(sb.toString());return;}String val = map[digits.charAt(index)-'2'];for(char ch:val.toCharArray()) {sb.append(ch);backtrack(digits,index+1);sb.deleteCharAt(sb.length()-1);}}
}

【总结】

1. 回溯递归通用模板,即用一个临时数组temp 来保存当前选出的子序列,使用cur 来表示当前位置的下标,在 dfs(cur, nums) 开始之前,[0,cur−1] 这个区间内的所有元素都已经被考虑过,而[cur,n] 这个区间内的元素还未被考虑。在执行 dfs(cur, nums) 时,我们考虑 cur 这个位置选或者不选,如果选择当前元素,那么把当前元素加入到temp 中,然后递归下一个位置,在递归结束后,应当把temp 的最后一个元素删除进行回溯;如果不选当前的元素,直接递归下一个位置。
List<List<Integer>> ans = new ArrayList<List<Integer>>();
List<Integer> temp = new ArrayList<Integer>();
public void dfs(int cur, int[] nums) {if (cur == nums.length) {// 判断是否合法,如果合法判断是否重复,将满足条件的加入答案if (isValid() && notVisited()) {ans.add(new ArrayList<Integer>(temp));}return;}// 如果选择当前元素temp.add(nums[cur]);dfs(cur + 1, nums);temp.remove(temp.size() - 1);// 如果不选择当前元素dfs(cur + 1, nums);
}
2.回溯相关题目

[Leetcode][第491题][JAVA][递增子序列][回溯][RK算法]
[Leetcode][第679题][JAVA][24点游戏][回溯][暴力]

[Leedcode][JAVA][第46题][全排列][回溯算法]
[Leetcode][第93题][JAVA][复原IP地址][剪枝][回溯]
[剑指offer]面试题第[38]题[JAVA][字符串的排列][回溯法]
[剑指offer][JAVA]面试题第[34]题[二叉树中和为某一值的路径][回溯]
[Leetcode][第17题][JAVA][电话号码的字母组合][回溯]
参考链接:https://leetcode-cn.com/problems/letter-combinations-of-a-phone-number/solution/dian-hua-hao-ma-de-zi-mu-zu-he-by-leetcode-solutio/

[Leetcode][第17题][JAVA][电话号码的字母组合][回溯]相关推荐

  1. [Leetcode][第78题][JAVA][子集][位运算][回溯]

    [问题描述][中等] [解答思路] 1. 位运算 复杂度 class Solution {List<Integer> t = new ArrayList<Integer>(); ...

  2. [Leetcode][第79题][JAVA][单词搜索][DFS][回溯]

    [问题描述][中等] [解答思路] 1. DFS繁琐版本 class Solution {public boolean exist(char[][] board, String word) {bool ...

  3. [Leetcode][第40题][JAVA][数组总和2][回溯][剪枝]

    [问题描述][中等] [解答思路] 1. 减法 import java.util.ArrayDeque; import java.util.ArrayList; import java.util.Ar ...

  4. [Leetcode][第216题][JAVA][数组之和3][回溯]

    [问题描述][中等] [解答思路] 回溯 剪树枝 当和超过n 或 个数超过k 1. 正向求和 优化前 class Solution {public List<List<Integer> ...

  5. [Leetcode][第81题][JAVA][N皇后问题][回溯算法]

    [问题描述][困难] [解答思路] 1. 主副对角线列 标记 复杂度 import java.util.ArrayDeque; import java.util.ArrayList; import j ...

  6. [Leetcode][第679题][JAVA][24点游戏][回溯][暴力]

    [问题描述][困难] [解答思路] 回溯 时间复杂度:O(1) 空间复杂度:O(1) class Solution {static final int TARGET = 24;static final ...

  7. [Leetcode][第39题][JAVA][组合总和][回溯][dfs][剪枝]

    [问题描述][中等] [解答思路] 1. 回溯 import java.util.ArrayDeque; import java.util.ArrayList; import java.util.De ...

  8. [Leetcode][第77题][JAVA][组合][回溯]

    [问题描述][中等] [解答思路] 1. 回溯 class Solution {List<List<Integer>> lists = new ArrayList<> ...

  9. [Leetcode][第889题][JAVA][根据前序和后序遍历构造二叉树][分治][递归]

    [问题描述][中等] [解答思路] copyOfRange class Solution {public TreeNode constructFromPrePost(int[] pre, int[] ...

最新文章

  1. Android Studio系列(二)使用Android Studio开发/调试整个android系统源代码(不定时更新)
  2. Feign深入学习(一)
  3. 关于返回一个整数数组中最大子数组的和的问题(续01)
  4. 通信原理实验(〇):音频信号的播放蒙特卡洛模拟
  5. 美化下拉框select箭头部分(不彻底)
  6. Redis 4.0深入持久化
  7. 报表工具Style Report报表打印功能
  8. jvisualvm 工具使用
  9. 局域网传输神器Snapdrop(电脑、手机均可)
  10. fiddler软件抓包工具超详细配置方法
  11. SDRAM控制器在连续读写数据时的优化
  12. Bagging 和 Boosting理解、区别与联系
  13. Win11启动修复无效怎么办
  14. 大数据十大“关键词”
  15. 香肠派对服务器维护时间,怎么解除香肠派对时间限制
  16. fseek函数的用法(用于设定指针位置)
  17. 手把手教你做主成分分析
  18. 聚类kmeans案例
  19. 索尼Xperia XZ1 Compact刷机后的问题,电量一直锁定20%,手机卡无信号无服务,相机拍照成纯绿色图片
  20. 计算机基础知识视频 银行考试,银行考试计算机基础知识试题及答案

热门文章

  1. UVa11988 Broken Keyboard(练习链表使用)
  2. hdu5468 Puzzled Elena
  3. Windows XP系统安装SQL Server 2005(开发版)图解
  4. [转贴]暴雪的霸王条款是否合理?
  5. vue router-link 添加点击事件
  6. Map<String,Object>接收参数,Long类型降级为Integer,报类型转换异常
  7. Docker中Maven私服的搭建
  8. Mac下Apache使用
  9. 从底层重学 Java 之两大浮点类型 GitChat连接
  10. Ant Design Pro 开发上手