题目:
A message containing letters from A-Z
is being encoded to numbers using the following mapping:
'A' -> 1
'B' -> 2
...
'Z' -> 26
Given an encoded message containing digits, determine the total number of ways to decode it.
For example,
Given encoded message "12"
, it could be decoded as "AB"
(1 2) or "L"
(12).
The number of ways decoding "12"
is 2.
提示:
这道题一开始想用backtracking构造出所有可能的decoding ways,然后统计个数,但是后来发现这样的解法不论是空间复杂度还是时间复杂度以及实现的难度都不太理想,一定还有更好的解决办法,没错,那就是动态规划了。
我们可以构造一个用于dp的vector,大小为输入string长度+1。其中第[0]个元素值为1,视为边界值,然后剩下的部分可以和输入string的每个字符都一一对应起来,其含义就是到该处时有多少种decoding方法。可想而知,第一个字符只要不是'0',就一定可以翻译,因此dp[1] = 1。
那么如何构造状态转移方程呢?假设我们处理到了s[i]的位置,那么首先单个s[i]的字符是可以翻译的,如果只翻译单个s[i],那么翻译的种类有dp[i]种。若s[i-1]与s[i]构成的字符串其值是小于等于26的,那么还可以再有dp[i-1]种翻译方法。总共dp[i]+dp[i-1]种(即到s[i-2]处有多少种翻译方法,结合上s[i-1,i];以及到s[i-1]有多少种翻译方法,结合上s[i])。
如果s[i-1]与s[i]构成的字符串其值大于26,那么显然就只能和s[i-1]处的翻译方法数量一样了。
另外还有一个特殊情况,即s[i]=0,因为单个0是没有办法翻译的,也就是说此时如果按照dp[i]的翻译方式(翻译到s[i-1],之后的字符串将以0打头)是不可行的,所以要将dp[i]赋值成0。
最后,实际上每次在进行状态转移时,我们最多只需要获取前两个状态,因此这里的空间复杂度还可以进一步从O(n)缩小到O(1),这里就不进一步展开了。上代码:
代码:
class Solution {
public:
int numDecodings(string s) {
int size = s.length();
if (size <= || s[] == '') {
return ;
}
vector<int> dp(size + , );
dp[] = dp[] = ;
for (int i = ; i < size; ++i) {
if (s[i] == '') {
dp[i] = ;
}
if (s[i - ] == '' || s[i - ] <= '' && s[i] <= '') {
dp[i + ] = dp[i] + dp[i - ];
} else {
dp[i + ] = dp[i];
}
}
return dp[size];
}
};