乘风破浪:LeetCode真题_017_Letter Combinations of a Phone Number
乘风破浪:LeetCode真题_017_Letter Combinations of a Phone Number
一、前言
如何让两个或者多个集合中的随机挑选的元素结合到一起,并且得到所有的可能呢?我们可能事先并不知道会用到哪几个集合,因此我们可以通过递归方法来解决。
二、Letter Combinations of a Phone Number
2.1 问题


2.2 分析和解决
通过上面的题目,我们基本上可以理解题意了,但是我们要怎么去处理呢,这个时候就需要用到递归算法了。
public class Solution {
private static final String[] KEYS = { "", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz" };
public List<String> letterCombinations(String digits) {
List<String> ret = new LinkedList<String>();
combination("", digits, 0, ret);
if(ret.get(0) == ""){
ret.clear();
}
return ret;
}
private void combination(String prefix, String digits, int offset, List<String> ret) {
if (offset >= digits.length()) {
ret.add(prefix);
return;
}
String letters = KEYS[(digits.charAt(offset) - '0')];
for (int i = 0; i < letters.length(); i++) {
combination(prefix + letters.charAt(i), digits, offset + 1, ret);
}
}
}
这里特别要注意,如果输入为空的情况下,我们的输出也必须为空,比如[]作为输入,[]也必须作为输出,而不是[""]。
当然我们也可以看看不同的递归写法:
import java.util.LinkedList;
import java.util.List; public class Solution { private String[] map = {
"abc",
"def",
"ghi",
"jkl",
"mno",
"pqrs",
"tuv",
"wxyz",
};
private List<String> result; // 存储最终结果
private char[] chars; // 保存去掉0,1字符的结果
private char[] curResult; // 存储中间结果
private int end = 0; // 字符数组中的第一个未使用的位置
private int handle = 0; // 当前处理的是第几个字符数字 /**
* 题目大意
* 给定一个数字串,返回数字上所有字符的所有组合,数字到字符的映射如上图所示。
* 注意: 尽管上面的结果以字符顺序排列的,你可以以任何顺序返回结果。
*
* 解题思路
* 一个数组保存数字和字的映射关用系,根据数字串的输入,找到对应的字符,组合结果。
*/
public List<String> letterCombinations(String digits) {
result = new LinkedList<>(); if (digits != null && digits.length() > 0) { chars = digits.toCharArray(); // 对字符串进行处理,去掉0和1
// 找第一个0或者1的位置
while (end < digits.length() && chars[end] != '0' && chars[end] != '1') {
end++;
} handle = end + 1;
while (handle < chars.length) {
if (chars[handle] != '0' && chars[handle] != '1') {
chars[end] = chars[handle];
end++;
}
handle++;
} curResult = new char[end];
// while结束后,end为有效字符的长度
handle = 0; // 指向第一个有效字符的位置 letterCombinations();
}
return result;
} private void letterCombinations() {
if (handle >= end) {
result.add(new String(curResult));
} else {
int num = chars[handle] - '2';
for (int i = 0; i < map[num].length(); i++) {
curResult[handle] = map[num].charAt(i);
handle++;
letterCombinations();
handle--;
}
}
}
}
上面的递归也是非常有意思的,大家可以仔细的玩味一下。

接下来我们再看一种自己构建队列的方式来处理的方法:
import java.util.*;
class Solution {
public List<String> letterCombinations(String digits) {
LinkedList<String> result = new LinkedList<>();
if(digits.isEmpty())
return result;
String[] mapping = new String[] {"0", "1", "abc","def","ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};
result.add(""); while(result.peek().length() != digits.length()) {
String remove = result.remove();
String map = mapping[digits.charAt(remove.length())-'0']; for(char c : map.toCharArray()) {
result.addLast(remove + c);
} } return result;
}
}
在这里定义了一个result队列,首先看看当前的指针所对应的元素的长度有没有达到给定的字符串的长度:
result.peek().length()
如果没有则继续,如果已经达到了,那么已经完成了遍历。
其次,最开始的元素出队列,它的长度就是下一个需要选择的给定字符串的位置,因此取出对应的数字,找到对应的字符串,然后将弹出的字符串和这些字符串拼接,入队列。
String map = mapping[digits.charAt(remove.length())-'0'];
for(char c : map.toCharArray()) {
result.addLast(remove + c);
}
依次这样遍历,知道满足条件,这样就得到了结果。
我们举一个例子:

同样的其实可以转换成堆栈来做,道理是一样的。
三、总结
当我们遇到问题的时候,不能解决的情况下一定要想到递归,动态规划这些方法,不然的话我们将无从做起,另外递归也有许多的技巧,都是可以转换成堆栈或者队列来解决的。
乘风破浪:LeetCode真题_017_Letter Combinations of a Phone Number的更多相关文章
- 乘风破浪:LeetCode真题_022_Generate Parentheses
乘风破浪:LeetCode真题_022_Generate Parentheses 一.前言 关于括号的题目,我们已经遇到过了验证正确性的题目,现在让我们生成合法的括号列表,怎么办呢?想来想去还是递归比 ...
- 乘风破浪:LeetCode真题_041_First Missing Positive
乘风破浪:LeetCode真题_041_First Missing Positive 一.前言 这次的题目之所以说是难,其实还是在于对于某些空间和时间的限制. 二.First Missing Posi ...
- 乘风破浪:LeetCode真题_040_Combination Sum II
乘风破浪:LeetCode真题_040_Combination Sum II 一.前言 这次和上次的区别是元素不能重复使用了,这也简单,每一次去掉使用过的元素即可. 二.Combination Sum ...
- 乘风破浪:LeetCode真题_039_Combination Sum
乘风破浪:LeetCode真题_039_Combination Sum 一.前言 这一道题又是集合上面的问题,可以重复使用数字,来求得几个数之和等于目标. 二.Combination Sum ...
- 乘风破浪:LeetCode真题_038_Count and Say
乘风破浪:LeetCode真题_038_Count and Say 一.前言 这一道题目,很类似于小学的问题,但是如果硬是要将输入和结果产生数值上的联系就会产生混乱了,因此我们要打破思维定势. ...
- 乘风破浪:LeetCode真题_037_Sudoku Solver
乘风破浪:LeetCode真题_037_Sudoku Solver 一.前言 这次我们对于上次的模型做一个扩展并求解. 二.Sudoku Solver 2.1 问题 2.2 分析与解决 这道题 ...
- 乘风破浪:LeetCode真题_036_Valid Sudoku
乘风破浪:LeetCode真题_036_Valid Sudoku 一.前言 有的时候对于一些基础知识的掌握,对我们是至关重要的,比如ASCII重要字符的表示,比如一些基本类型的长度. 二.Valid ...
- 乘风破浪:LeetCode真题_035_Search Insert Position
乘风破浪:LeetCode真题_035_Search Insert Position 一.前言 这次的问题比较简单,也没有限制时间复杂度,但是要注意一些细节上的问题. 二.Search Insert ...
- 乘风破浪:LeetCode真题_034_Find First and Last Position of Element in Sorted Array
乘风破浪:LeetCode真题_034_Find First and Last Position of Element in Sorted Array 一.前言 这次我们还是要改造二分搜索,但是想法却 ...
随机推荐
- ubuntu下终端路径显示的修改
环境:ubuntu16.04 ubuntu在默认情况下是显示绝对路径的,进入目录过长的时候让人感觉很不舒服,现在修改成只显示当前目录 vim ~/.bashrc 找到这句 # If this is a ...
- redis数据类型(四)list类型
一.list类型 list是一个链表结构,可以理解为一个每个子元素都是 string 类型的双向链表. 主要功能是push.pop.获取一个范围的所有值等. 操作中key理解为链表的名字. 二.Lis ...
- 如何封装一个Cookie库
由Cookie详解我们已经了解到了Cookie是为了实现有状态的基于HTTP协议的应用而存在的.一个Cookie的由以下几个部分组成: //设置cookie的格式和Set-Cookie头中使用的格式一 ...
- 经典SQL分页语句
select top pageSize, * from (SELECT row_number() over(order by id desc) as rownumber,*FROM tb1) A wh ...
- 百度搜索URL参数含义
序号 参数 含义 1 tn 搜索框所属网站.比如 tn=sitehao123,就是 http://www.hao123.com/ 左上那个搜索框(指通过什么方式到达百度首页搜索界面;) 2 s?wd ...
- [javaEE] 控制浏览器缓存资源
浏览器有默认的缓存机制,不同的浏览器,缓存头是不一样的 设置编码,调用setContentType()方法,参数:”text/html;charset=utf-8” 关闭缓存,调用setHeader( ...
- Kinect1驱动 PCL OpenCV ROS 安装
1. OpenCV安装 1)在终端安装依赖项sudo apt-get install build-essential libgtk2.0-dev libjpeg-dev libtiff4-dev li ...
- 关于springmvc中常用的注解,自己也整理一下
1.@Controller 在springMVC中@controller主要用在控制层的类上,之前只知道用注解开发的时候必须加一个@controller ,今天看了别的大佬整理的才知道为什么这么用,控 ...
- linux ubuntu 学习总结(day01)基本命令学习
个人总结,请尊重版权,转载请在显眼地方注明出处:https://www.cnblogs.com/sunshine5683/p/9948259.html day(01)基本命令学习 今天开始自学linu ...
- J2EE企业级应用架构
一.企业级应用架构解析 应用特点 多环境多系统的交互 海量数据.高并发[用户访问量].高TPS[每秒吞吐量] 安全等级高 自动化集群管理 架构原则 CAP原则(一致性[数据变动要同步].可用性[随着数 ...