题目:

Given a digit string, return all possible letter combinations that the number could represent.

A mapping of digit to letters (just like on the telephone buttons) is given below.

Input:Digit string "23"
Output: ["ad", "ae", "af", "bd", "be", "bf", "cd", "ce", "cf"].

Note:
Although the above answer is in lexicographical order, your answer could be in any order you want.

代码:

class Solution {
public:
vector<string> letterCombinations(string digits) {
vector<string> ret;
if ( digits.empty() ) return ret;
map<int, string> digitLetters;
digitLetters[] = "";
digitLetters[] = "";
digitLetters[] = "abc";
digitLetters[] = "def";
digitLetters[] = "ghi";
digitLetters[] = "jkl";
digitLetters[] = "mno";
digitLetters[] = "pqrs";
digitLetters[] = "tuv";
digitLetters[] = "wxyz";
vector<int> letterBegin(,);
int index = ;
string tmp;
Solution::combine(ret, tmp, index, digits, digitLetters, letterBegin);
return ret;
}
static void combine(
vector<string>& ret,
string& tmp,
int index,
string& digits,
map<int,string>& digitLetters,
vector<int>& letterBegin)
{
if (index==digits.size())
{
ret.push_back(tmp);
return;
}
int curr = digits[index]-'';
string letters = digitLetters[curr];
for ( int i = ; i < letters.size(); ++i )
{
letterBegin[curr] = i;
tmp.push_back(letters[i]);
Solution::combine(ret, tmp, index+, digits, digitLetters, letterBegin);
tmp.erase(tmp.end()-);
}
}
};

tips:

上述的代码是AC的。思路也就是常规dfs的思路:

1. 终止条件是层级到了digits的长度

2. 每一层递归相当于根据该位置的数字选一个字母

但是,个人感觉这道题的题干要求没说清楚;既然是letter combinations,而不是letter permutation,那么“ac”和“ca”就应该算一个combination,但是OJ之后发现默认“ac”和“ca”算两个。加入“ac”和“ca”算一个,有没有解法呢?代码如下:

class Solution {
public:
vector<string> letterCombinations(string digits) {
vector<string> ret;
if ( digits.empty() ) return ret;
map<int, string> digitLetters;
digitLetters[] = "";
digitLetters[] = "";
digitLetters[] = "abc";
digitLetters[] = "def";
digitLetters[] = "ghi";
digitLetters[] = "jkl";
digitLetters[] = "mno";
digitLetters[] = "pqrs";
digitLetters[] = "tuv";
digitLetters[] = "wxyz";
vector<int> letterBegin(,);
int index = ;
string tmp;
Solution::combine(ret, tmp, index, digits, digitLetters, letterBegin);
return ret;
}
static void combine(
vector<string>& ret,
string& tmp,
int index,
string& digits,
map<int,string>& digitLetters,
vector<int>& letterBegin)
{
if (index==digits.size())
{
ret.push_back(tmp);
return;
}
int curr = digits[index]-'';
string letters = digitLetters[curr];
for ( int i = letterBegin[curr]; i < letters.size(); ++i )
{
letterBegin[curr] = i;
tmp.push_back(letters[i]);
Solution::combine(ret, tmp, index+, digits, digitLetters, letterBegin);
tmp.erase(tmp.end()-);
}
}
};

tips:

多维护一个letterBegin的数组,标记“当前的组合中,某个数字对应的字母序列中应该从第几个字母开始取”。

如果输入是“22”,那么给出的解集就是:

aa

ab

ac

bb

bc

cc

从这个例子可以看出来,算法的原理就是维护某一个数字对应字母序列:如果数字重复出现,那么对应的字母要保证字典序递增,这样就不会用重复的。

可惜题目并不是这么要求的。

===============================================

既然题目要求简单了,则再追求一个迭代的解法。AC的代码如下:

class Solution {
public:
vector<string> letterCombinations(string digits) {
vector<string> ret;
if (digits.empty()) return ret;
ret.push_back("");
map<int, string> digitLetters;
digitLetters[] = "abc";
digitLetters[] = "def";
digitLetters[] = "ghi";
digitLetters[] = "jkl";
digitLetters[] = "mno";
digitLetters[] = "pqrs";
digitLetters[] = "tuv";
digitLetters[] = "wxyz";
for ( size_t i = ; i < digits.size(); ++i )
{
int curr = digits[i]-'';
string letters = digitLetters.find(curr)==digitLetters.end() ? "" : digitLetters[curr];
vector<string> tmp = ret;
ret.clear();
for ( size_t j = ; j < tmp.size(); ++j )
{
for ( size_t k = ; k < letters.size(); ++k )
{
string ori = tmp[j];
ori += letters[k];
ret.push_back(ori);
}
}
}
return ret;
}
};

完毕。

======================================

第二次过这道题,用dfs过的。注意如果原来的digits==""返回的也是空。

class Solution {
public:
vector<string> letterCombinations(string digits)
{
map<char, string> digit_letters;
digit_letters[''] = "abc";
digit_letters[''] = "def";
digit_letters[''] = "ghi";
digit_letters[''] = "jkl";
digit_letters[''] = "mno";
digit_letters[''] = "pqrs";
digit_letters[''] = "tuv";
digit_letters[''] = "wxyz";
vector<string> ret;
if ( digits=="" ) return ret;
vector<char> tmp;
Solution::dfs(ret, digits, tmp, digit_letters);
return ret;
}
static void dfs(
vector<string>& ret,
string digits,
vector<char>& tmp,
map<char,string>& digit_letters)
{
if ( tmp.size()==digits.size() )
{
ret.push_back(string(tmp.begin(),tmp.end()));
return;
}
int index = tmp.size();
for ( int i=; i<digit_letters[digits[index]].size(); ++i )
{
tmp.push_back(digit_letters[digits[index]][i]);
Solution::dfs(ret, digits, tmp, digit_letters);
tmp.pop_back();
}
}
};

最新文章

  1. linux 学习6 软件包安装
  2. S2--《深入.NET平台和C#编程》
  3. angular_$attrs
  4. groovy-语句
  5. Ora中select某时间段记录sql语句
  6. Android 切横竖屏时走的生命周期方法?222
  7. java历史版本下载地址
  8. ehcarts 四川地图
  9. 2017.11.13 flex 布局相关问题
  10. 在Ubuntu虚拟机上安装DVWA
  11. 网易云首席安全架构师谈安全新形势:DDOS两三天,游戏玩家数从几万降到几百
  12. 如何掌握 Kubernetes ?系统学习 k8s 的大纲一份
  13. iframe中的历史记录问题汇总及解决方案[转]
  14. weblogic 12C 安全加强:更改Console及管理端口
  15. css继承和层叠
  16. express函数参数之next
  17. 20155209 2016-2017-2 《Java程序设计》第二周学习总结
  18. 【Machine Learning】监督学习、非监督学习及强化学习对比
  19. 浅析C#中 ConcurrentDictionary的实现
  20. WARNING: at drivers/gpio/gpiolib.c:101 gpio_ensure_requested+0x5c/0x118()

热门文章

  1. fstab 解析
  2. pg中的非varchar类型的模糊搜索
  3. java Vamei快速教程16 RTTI
  4. spring-autowire机制
  5. 【BZOJ2243】[SDOI2011] 染色(树链剖分)
  6. 用ComboBox控件制作浏览器网址输入框
  7. Shuffle Cards
  8. Java反射得到属性的值和设置属性的值
  9. 如何着手学习一个新的PHP框架
  10. Wordpress菜单函数wp_nav_menu各参数详解及示例