leetcode 面试题91.解码方法

2020/3/17  打卡

题目

一条包含字母 A-Z 的消息通过以下方式进行了编码:

'A' -> 1
'B' -> 2
...
'Z' -> 26
给定一个只包含数字的非空字符串,请计算解码方法的总数。

示例 1:
输入: "12"
输出: 2
解释: 它可以解码为 "AB"(1 2)或者 "L"(12)。

示例 2:
输入: "226"
输出: 3
解释: 它可以解码为 "BZ" (2 26), "VF" (22 6), 或者 "BBF" (2 2 6) 。

思路

这道题和爬楼梯的那道题的思路很像,都是最后一种状态由前两种状态转化而来。(使用动态规划,使用类似楼梯的方法解决即可)
所以直接使用动态规划进行情况搜索即可,状态值表示解码方法的总数。

 1.我们知道最后一个字母有可能是由一个数字串或两个数字串转化而来,例如'12312'这个数字串,转化成字母串最后一位可以是'2'转成B,也可能是'12'转成L

 2.设数字串S的前i个数字解密成字母串有dp[i]种方式:那么就有dp[i] = dp[i-1] + dp[i-2]
     dp[i-1]表示最后一个数字解密成一个字母
     dp[i-2]表示最后两个数字解密成一个字母

 3.由于这里存在很多特殊情况:
     例如dp[i-2]表示最后两个数字解密成一个字母的情况必须要至少有两个数字
     例如最后两个字母大于26,像55、01等是不存在dp[i-2]这种情况的

所以,像下面的拆开写状态转移方程,分别判断是很好的思路,见代码。

代码

def DP_fun(s):
    size=len(s)

    # 边界条件
    if size==0:
        return 0

    #创建 状态,表示 到当前位置,可以有的解码方法总数
    dp=[0]*(size+1)

    # 状态初始化
    dp[0]=1

    #  进行状态转移操作
    for i in range(1,size+1):
        #——————————  接下来对于每个位置,都做了 dp[i] = dp[i-1] + dp[i-2]组合的相加 ——————————————————

        t = int(s[i - 1])
        # 最后一个数字解密成一个字母。  先加上  前一个包装成dp[i-1]的情况。
        if t >= 1 and t <= 9:
            dp[i] += dp[i - 1]

        # 下面这种情况至少要有两个字符。 再加上  前两个可以包装成dp[i-2]的情况(需要组合数在 0-26范围内。)。
        if i >= 2:
            t = int(s[i - 2]) * 10 + int(s[i - 1])
            if t >= 10 and t <= 26:
                # 最后两个数字解密成一个字母
                dp[i] += dp[i - 2]

    # 最后一个位置上的统计 ,就是整体对整个字符串的 编码方法总数上的统计。
    return dp[-1]

 

你可能感兴趣的:(马飞飞的刷题日记)