题目:解码方法数

难度:Medium

题目内容

A message containing letters from A-Z is being encoded to numbers using the following mapping:

'A' -> 1
'B' -> 2
...
'Z' -> 26

Given a non-empty string containing only digits, determine the total number of ways to decode it.

翻译

一个包含A-Z字母的消息被编码成数字,使用以下映射:

“A” - > 1

“B”- > 2

……

“Z”- > 26

给定一个包含数字的非空字符串,确定解码它的总方法数。

我的思路:一开始想用递归但是边界问题太多,就放弃了。

      single 从0开始(每个数组单独解码),count从1开始,如果s[i-1]s[i]二者组成的数字在(0,26 ] 范围内,那就count+;

      且遇见0的时候,一开始的单独编码就不存在了,且少了一次组合的机会,例如【110】 中【11】和【0】不能组合

        所以此时 single = 0;count - 1

我的代码

     public int numDecodings(String s) {
if (s.isEmpty() || s.charAt(0) == '0')
return 0; int count = 0;
int single = 1;
for (int i = 1; i < s.length(); i++) {
if (s.charAt(i) == '0'){
count = s.length() > 2 ? count-1 : count;
single = 0;
}
int x = Integer.parseInt(s.substring(i-1, i+1));
if ( x<= 26 && x > 0) {
count += 1;
}
}
return (count + single) > 0 ? (count + single) : 0;
}

结果:214 / 258 test cases passed.

Input:"1212"
Output:4
Expected:5

编码过程中的问题

1、最初 single 和 count 没有分开计算;

2、第9行,当只有两个的时候,此时count 不需要 -1 ,例如【10】;

3、这个思路还是有问题的,例如【1212】中少计算了【12】【12】这个组合

答案代码

 public class Solution {
public int numDecodings(String s) {
int n = s.length();
if (n == 0) return 0; int[] memo = new int[n+1];
memo[n] = 1;
memo[n-1] = s.charAt(n-1) != '0' ? 1 : 0; for (int i = n - 2; i >= 0; i--)
if (s.charAt(i) == '0') continue;
else memo[i] = (Integer.parseInt(s.substring(i,i+2))<=26) ? memo[i+1]+memo[i+2] : memo[i+1]; return memo[0];
}
}

答案思路

假设所有的数字都有效,且左右两两相邻数之间也有效(1到26),那么有如下规律

   numDecodings(s) = numDecodings(s.substring(1)) + numDecodings(s.substring(2)) --------------------------------------------------式(1)

规律显而易见,斐波那契数列。不过是反过来的(从字符串后面往前)

当然,那只是假设,也要考虑特殊情况:

【使用一个数组(大小为len+1)从后往前记录对应的数字“出现”后解码数的增量

1、当前指针所指字符为0

  此时此字符无法解码,所以式(1)中的前者就只能为0,后者也为0

  例如【023】,substring(1)——【0】|【23】,截掉的【0】不能解析,所以此组合无效

         substring(2)——【02】|【3】,截掉的【02】不能解析,所以此组合无效

  所以0数字出现后,解码数的增量为0。

2、当前字符的值是有效的(大于0),但是当前字符与右边字符组合的数字无效(大于26)

  相当于式(1)中的后者=0

  例如【3212】,substring(1)——【3】|【212】,截掉的【3】能解析,所以其值为【212】的解码数

         substring(2)——【32】|【12】,截掉的【32】不能解析,所以此组合无效\

递归实现:

     public int numDecodings(String s) {
if (s.isEmpty()) {
return 1;
} if (s.charAt(0) == '0') {
return 0;
} if (s.length() == 1)
return 1; int sub1 = numDecodings(s.substring(1));
int sub2 = 0; if (Integer.parseInt(s.substring(0,2)) <= 26) {
sub2 = numDecodings(s.substring(2));
}
return sub1 + sub2;
}

此方法会在最后几个用例(很长)运行超时。

所以答案采用了迭代的方式进行。

优化:既然是斐波纳数列,那么就能使用双值迭代的方式取代用一个数组进行记录。

     public int numDecodings(String s) {
if (s.isEmpty() || s.charAt(0) == '0') {
return 0;
}
int sub1 = 1;
int sub2 = 1;
for (int i = s.length() - 2; i > -1; i--) {
if (s.charAt(i+1) == '0') {
sub1 = 0;
}
if (Integer.parseInt(s.substring(i,i+2)) <= 26) {
sub1 = sub1 + sub2;
sub2 = sub1 - sub2;
} else {
sub2 = sub1;
}
}
return sub1;
}

第11行其实还可以使用字符进行判断以降低时间和空间复杂度:

   s.charAt(i) == '1' || s.charAt(i) == '2' && s.charAt(i+1) - '0' <= 6

LeetCode第[91]题(Java):Decode Ways(解码方法数)的更多相关文章

  1. Leetcode 91. Decode Ways 解码方法(动态规划,字符串处理)

    Leetcode 91. Decode Ways 解码方法(动态规划,字符串处理) 题目描述 一条报文包含字母A-Z,使用下面的字母-数字映射进行解码 'A' -> 1 'B' -> 2 ...

  2. LeetCode第[18]题(Java):4Sum 标签:Array

    题目难度:Medium 题目: Given an array S of n integers, are there elements a, b, c, and d in S such that a + ...

  3. LeetCode第[1]题(Java):Two Sum 标签:Array

    题目: Given an array of integers, return indices of the two numbers such that they add up to a specifi ...

  4. LeetCode第[46]题(Java):Permutations(求所有全排列) 含扩展——第[47]题Permutations 2

    题目:求所有全排列 难度:Medium 题目内容: Given a collection of distinct integers, return all possible permutations. ...

  5. LeetCode第[1]题(Java):Two Sum (俩数和为目标数的下标)——EASY

    题目: Given an array of integers, return indices of the two numbers such that they add up to a specifi ...

  6. [LeetCode] 91. Decode Ways 解码方法

    A message containing letters from A-Z is being encoded to numbers using the following mapping: 'A' - ...

  7. [LeetCode] Decode Ways 解码方法

    A message containing letters from A-Z is being encoded to numbers using the following mapping: 'A' - ...

  8. [LintCode] Decode Ways 解码方法

    A message containing letters from A-Z is being encoded to numbers using the following mapping: 'A' - ...

  9. [leetcode]91. Decode Ways解码方法

    A message containing letters from A-Z is being encoded to numbers using the following mapping: 'A' - ...

随机推荐

  1. 用仿ActionScript的语法来编写html5——第一篇,显示一张图片

    第一篇,显示一张图片 一,代码对比 as代码: public var loader:Loader; public function loadimg():void{ loader = new Loade ...

  2. Jmeter(八)Jmeter监控tomcat

    1.配置tomcat的配置文件conf/tomcat-users.xml 2. 在“线程组”上右键“添加”--“配置元件”--“HTTP授权管理器”,这里主要是让JMeter能够通过Tomcat的基本 ...

  3. Spark2.0机器学习系列之8:多类分类问题(方法归总和分类结果评估)

    一对多(One-vs-Rest classifier) 将只能用于二分问题的分类(如Logistic回归.SVM)方法扩展到多类. 参考:http://www.cnblogs.com/CheeseZH ...

  4. 深度分析ORACLE热点块问题

    1.热点块的定义 数据库的热点块,从简单了讲,就是极短的时间内对 少量数据块进行了过于频繁的访问.定义看起来总是很简单的,但实际在数据库中,我们要去观察或者确定热点块的问题,却不是那么简单了.要深刻地 ...

  5. Linux系统——引导过程与服务控制

    一.Linux开机启动原理(十步) (1)开机自检BIOS 开机检测,主板检测 (2)MBR引导 硬盘512字节 (3)GRUB菜单 操作系统菜单 (4)加载内核(kernel) 启动操作系统核心,根 ...

  6. HDU - 2732 Leapin' Lizards (拆点最大流)

    题意:有N*M的矩形,每个格点有一个柱子,每根柱子有高度c,允许蜥蜴经过这根柱子c次,开始有一些蜥蜴在某些柱子上,它们要跳出这个矩形,每步最大能跳d个单位,求最少有多少蜥蜴不能跳出这个矩形. 分析:转 ...

  7. ElasticSearch recovery过程源码分析

    [ES版本] 5.5.0 [分析过程] 找到Recovery有6种状态 public class RecoveryState implements ToXContent, Streamable { p ...

  8. selenium实现excel文件数据的读、写

    在进行软件测试或设计自动化测试框架时,一个不可避免的过程就是: 参数 化,在利用 python 进行自动化测试开发时,通常会使用 excel 来做数据管 理,利用 xlrd.xlwt 开源包来读写 e ...

  9. Django学习笔记之Django的url反向解析

    0x00 URL反向解析和三种不同的反向解析方式 Django中提供了关于URL的映射的解决方案,可以做两个方向的使用: 1.普通解析过程:由客户端的浏览器发起一个url请求,Django根据URL解 ...

  10. JavaScript浮点运算,小数点精度

    math.js JavaScript浮点运算,小数点精度 // JavaScript Document //数学函数 // 浮点数加法运算 function floatAdd(arg1, arg2) ...