leadcode的Hot100系列--17. 电话号码的字母组合--回溯的另一种想法的应用
提交leetcode的时候遇到了问题,一直说访问越界,但仔仔细细检查n多遍,就是检查不出来。
因为我用到了count全局变量,自加一来表明当前数组访问的位置,
后来突然想到,是不是在leetcode在运行测试用例的时候,是连续测试的,用的同一个上下文,这样的话,就没有对这个全局变量清零……
果然,清零之后就可以了……已经3:47了,这里先上代码,明天再详细说吧……
今天更新一下这道题的思路。
可以先参考一下之前的两篇文章,循序渐进,好理解一些:
leadcode的Hot100系列--78. 子集--位运算
leadcode的Hot100系列--78. 子集--回溯
在 子集--回溯
的文章里面,介绍了一下数字的排列组合,用01来表示对应的数字是否存在。
如果我们还是按照这个思路,但是换一个想法呢?
0、1是不是本身就可以代表着字符串?
对应排列出来的000\001\010 ... 是不是就是相当于:
我需要一个数字组合,组合需要三位数,每一位的数字要么是0,要么是1。
这么一想,是不是就与题目一致了:
我需要一个字母组合,组合的位数就是输入的字符串长度,每一位的字母是对应的几个字母中的某一个。
对,就是这么想的,比如,输入“89”,就说明,字母组合的位数是两位,第一位字母是'tuv'里面的一个,第二位字母是'wxyz'里面的一个。
这里再看下之前上一篇中回溯的代码:
void backtrack (int t)
{
if (t == level)
show();
else
for (int i=0;i<=1;i++)
{
y[t]=i;
backtrack(t+1);
}
}
重点来了!!!!
- 第三行的level表示层数,也就是遍历的深度,也就是组合所需要的位数,当需要两位字母的时候,就只要两层。
- 第六行的for循环,表示了每一层的选项,之前因为只需要表示存在和不存在,所以只需要用0和1就够了,但在这里,是由数字对应的字符串的某一个,例如如果数字是8,则对应的选项就是't'和'u'和'v'。
所以,当输入为“89”的时候,就可以生成这样一种树:
控制了树的层数和树的分支(分支就是可选项)之后,就可以完成所有组合。
char table[][5] = {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};
char level = 0;
char *p[8]; // 指向数字对应的字符串,例如,当输入数字为"89"时,p[0]为"tuv",p[1]为"wxyz"。
char len[8]; // 对应上面p存储的字符串的长度,例如,当输入数字为"89"是,len[0]=3,len[1]=4。
char **out; //二维数组,是最终输出
int count = 0; // 用来记录当前已经生成了几个组合,对应着out数组的行坐标
char y[8] = {0}; // 记录每一次的组合结果
void backtrack(int level_now)
{
if (level_now == level)
{
memcpy(out[count], y, level); // 把这次组合结果拷贝到out数组中。这里为什么需要用一个y数组来记录组合结果,然后拷贝到out中呢?大家可以自己想一想
count ++; // 完成一个字符串
return;
}
for (int i=0; i<len[level_now]; i++)
{
y[level_now] = p[level_now][i];
backtrack(level_now+1);
}
return;
}
char ** letterCombinations(char * digits, int* returnSize){
level = strlen(digits); // 遍历的层数
*returnSize = 0;
if (0 == level) return NULL;
*returnSize = 1;
for(int i=0; i<level; i++)
{
p[i] = table[digits[i]-'0']; // 对p数组进行赋值
len[i] = strlen(p[i]);
if (len[i] == 0)
{
*returnSize = 0;
return NULL;
}
*returnSize *= len[i]; // 计算总共有多少个组合
}
out = (char **)calloc(*returnSize, sizeof(char *)); // 先分配行指针
if (NULL == out) return NULL;
for (int i=0; i<*returnSize; i++)
{
out[i] = (char *)calloc(1, sizeof(char) * (level+1)); // 再分配每个行指针的内容,因为字符串后面需要一个结束符'\0',所以这里需要level+1
if (NULL == out[i]) return NULL;
}
backtrack(0);
count = 0; // 这里很重要!很重要!!很重要!!!
return out;
}
leadcode的Hot100系列--17. 电话号码的字母组合--回溯的另一种想法的应用的更多相关文章
- [LeetCode] 17. 电话号码的字母组合 ☆☆☆(回溯) ###
描述 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合. 给出数字到字母的映射如下(与电话按键相同).注意 1 不对应任何字母. 示例: 输入:"23"输出:[&q ...
- [LeetCode] 17. 电话号码的字母组合(回溯)
题目 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合. 给出数字到字母的映射如下(与电话按键相同).注意 1 不对应任何字母. 示例: 输入:"23" 输出:[& ...
- leadcode的Hot100系列--64. 最小路径和--权值最小的动态规划
如果这个: leadcode的Hot100系列--62. 不同路径--简单的动态规划 看懂的话,那这题基本上是一样的, 不同点在于: 1.这里每条路径相当于多了一个权值 2.结论不再固定,而是要比较不 ...
- Leetcode之回溯法专题-17. 电话号码的字母组合(Letter Combinations of a Phone Number)
[Leetcode]17. 电话号码的字母组合(Letter Combinations of a Phone Number) 题目描述: 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组 ...
- Java实现 LeetCode 17 电话号码的字母组合
17. 电话号码的字母组合 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合. 给出数字到字母的映射如下(与电话按键相同).注意 1 不对应任何字母. 示例: 输入:"23& ...
- leetcode(js)算法之17电话号码的字母组合
给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合. 给出数字到字母的映射如下(与电话按键相同).注意 1 不对应任何字母 示例: 输入:"23" 输出:[" ...
- [LeetCode] 17. 电话号码的字母组合
题目描述:https://leetcode-cn.com/problems/letter-combinations-of-a-phone-number/ 题目描述: 给定一个仅包含数字 2-9 的字符 ...
- Leetcode题库——17.电话号码的字母组合
@author: ZZQ @software: PyCharm @file: letterCombinations.py @time: 2018/10/18 18:33 要求:给定一个仅包含数字 2- ...
- leetcode题目17.电话号码的字母组合(中等)
题目描述: 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合. 给出数字到字母的映射如下(与电话按键相同).注意 1 不对应任何字母. 示例: 输入:"23"输出: ...
随机推荐
- WPF里的一些Effect特效
原文:WPF里的一些Effect特效 Blend的特效都在Microsoft.Expression.Media.Effects里,用之前添加一下引用. 可以在前台选中对象后直接点击Effect新建一种 ...
- Cocos2d-x 脚本语言Lua介绍
Cocos2d-x 脚本语言Lua介绍 本篇博客记录Lua学习.学习来自eoe论坛,Lua语言开发Cocos2d-x游戏入门视频教程,猛戳下面地址: http://www.eoeandroid.com ...
- WPF特效-粒子动画
原文:WPF特效-粒子动画 WPF实现泡泡龙小游戏效果. /// -Ball to Ball Collision - Detection and Handling /// http:// ...
- 早期malloc分配时,如果内存耗尽分配不出来,会直接返回NULL。现在分配不出来,直接抛出异常(可使用nothrow关键字)
今天和同事review代码时,发现这样的一段代码: Manager * pManager = new Manager(); if(NULL == pManager) { //记录日志 return f ...
- Post utf-8 请求
/// <summary> /// POST请求与获取结果 /// </summary> public static string HttpPost(string Url, s ...
- VC中出现“烫”和“屯”的原因(栈区的每一个字节都被0xCC填充了,也就是int 3h的机器码,动态分配的堆,VC的Debug用0xCD填充堆的空间,就出现了“屯”)
相信经常用VC的朋友对屏幕输出的一大堆“烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫烫”不会陌生,但是也许会很奇怪,为什么会出现“烫”字呢?莫非改程序导致系统运行缓慢,发热过高???非也!下面让我解释 ...
- Image Caption论文合辑2
说明: 这个合辑里面的论文不全是Image Caption, 但大多和Image Caption相关, 同时还有一些Workshop论文. Guiding Long-Short Term Memory ...
- Win8Metro(C#)数字图像处理--2.28图像乘法运算
原文:Win8Metro(C#)数字图像处理--2.28图像乘法运算 [函数名称] 图像乘法函数MultiplicationProcess(WriteableBitmap src, Writea ...
- Android基础开发入门(一)
前言:我学了一年多的C#(从学编程算起,也没有两年,我现在大二下),中间也一直在学WP开发,虽然技术不咋地,很渣渣,但微软在Build大会上宣布的策略让我觉得有必要学习一下安卓开发了.关于微软的策略, ...
- VS2008发布项目“发布失败”,没有提示错误
VS2008发布项目时发布失败,但是没有提示任何的错误. 解决方法: 组合键”Ctrl+Alt+O”; 根据这个我知道了,是因为我更改了文件名,发布时找不到导致的,然后在资源管理器中找到那一项,删除或 ...