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.
思路:
这是一个动态规划的问题。考察输入串中第i,i-1,i-2的三个字符。设decode_ways[i]为到第i个字符为止,找到的解码个数,s[i]为输入串中第i个字符。那么s[i-1]和s[i]可以有几种可能:
- s[i],s[i-1]分别可以解码,同时s[i-1]s[i]组成的二位数也可以解码。这样,decode_ways[i]应该是decode_ways[i-1]和decode_ways[i-2]的和。例如"221",解码的数字等于”22“解码的数字(多余的1可以单独解码)与”2“解码的数字(多余的”21“也可以单独解码)之和。
- s[i]不能单独解码(为0),那么必须和s[i-1]组成一个部分,decode_ways[i]只能等于decode_ways[i-2]。
- s[i-1]s[i]不能解码,但是s[i]可以解码。那么decode_ways[i-2]将无法贡献,解码的数字decode_ways[i]等于decode_ways[i-1]。
- 其他情况(包括几种不可能存在的情况,如s[i-1]s[i]可以解码但是s[i],s[i-1]都不可以解码),decode_ways[i]只能等于0。
由此写出状态转移方程和代码。
状态转移方程:
1.如果S[i] == '0',如果S[i-1]存在且为'1'或者'2',F[i] = F[i-1],否则无解;
2.如果S[i] != '0' 如果S[i-1]=='1',F[i] = F[i-1] + F[i-2](例如"xxxxx221",可以是"xxxxx22"+"1",也可以是"xxxx2"+"21"); 如果S[i-1]=='2',当S[i] <= '6'时,F[i] = F[i-1] + F[i-2] (最大的Z为"26","27""28""29"不存在),当S[i] > '6'时,F[i] = F[i-1] (例如"xxxxxx28",只能是"xxxxxx2" + "8")。代码:
class Solution {public: int numDecodings(string s) { int n = s.length(); if (n<1) { return 0; } vector dp(n+1, 0); dp[0] = 1; for (int i=1; i
参考:
1. http://blog.csdn.net/magisu/article/details/16942907
2. http://www.tuicool.com/articles/JVrMBr