把数字翻译成字符串:最直观的想法是,一开始想,首先使用dp[i]表示前i个字符串的翻译结果数,初始化均为1,因为字符串一个字符一个字符翻译则结果数为1,然后从第二个元素开始遍历,如果当前字符和前一个字符组成的数在10~26之间,则会出现两种翻译结果,一种是一个字符一个字符,另一种是当前字符和前一个字符,dp[i]=2*dp[i-1],反之则直接翻译,dp[i]=dp[i-1],最后返回dp[nums.size()-1]即可。(出错)
分析:但是上述这样想出错了,那么肯定是递推公式有些许问题,不过动态规划本身就有点复杂,错了也没关系,肯定是某些方面没有考虑全面,找找原因再接再厉好啦。根据上述描述,我们不妨将一个字符一个字符称为直接译码,当前字符和前一个字符称为组合译码,两种结果的翻译结果应该是dp[i]=dp[i-1]+dp[i-2],而不是dp[i]=2*dp[i-1],因为直接译码是dp[i]=dp[i-1],而组合译码是当前和前一个组合,于是是截止前前一个的结果,即dp[i]=dp[i-2]。同时,注意,字符串中可能存在数字0,而数字0是不能被编码的,所以不应该是10~26,而应该是排除10和20,得到11~19和21~26,因为10和20算了两种,分别是1和0还有10、2和0还有20,但是0不能被编码,要考虑这一点,10和20只有一种编码,即10和20,那么dp[i]=dp[i-2]。
总结:综上所述,一共有三种情况,根据当前字符和前一个字符的组合情况:第一种是10和20,则dp[i]=dp[i-2];第二种是11~19或者21~26,则dp[i]=dp[i-1]+dp[i-2];第三种是其他情况,则dp[i]=dp[i-1]。在终止条件中需要考虑如果当前字符为0而前一个字符不为1且不为2时需要返回0。代码如下,虽然有点冗余,但是十分清晰。(成功不是一蹴而就的,进一寸有进一寸的欢喜)
int solve(string nums) { //空字符串 if(nums.size()==0) return 0; //字符串0 if(nums=="0") return 0; //字符串10或者20 if(nums=="10"||nums=="20") return 1; //当0前面不是1或者2时无法译码 for(int i=1;i<nums.size();i++) { if(nums[i]=='0') if(nums[i-1]!='1'&&nums[i-1]!='2') return 0; } //dp[i]表示以i为结尾的翻译结果种数 vector<int> dp(nums.size(),1); for(int i=1;i<nums.size();i++) { if(nums[i]=='0') { if(nums[i-1]=='1'||nums[i-1]=='2') { if(i==1) dp[i]=1; //数字以10或者20开头 else dp[i]=dp[i-2]; //数字中存在10或者20 } } else if(nums[i-1]=='1'||(nums[i-1]=='2'&&nums[i]>='1'&&nums[i]<='6')) { if(i==1) dp[i]=2; //数字是11~19或者21~26开头 else dp[i]=dp[i-1]+dp[i-2]; //组合译码 } else dp[i]=dp[i-1]; //直接译码 } return dp[nums.size()-1]; }