1. 题目

这是《剑指offer》上的一道题,刚开始觉得这是一道挺简单的题目,后来发现自己太年轻了,考虑的因素太少了,思考了而是分钟还是无从下手,看了作者的思路深深被他折服了,题目如下:

请实现一个函数用来判断字符串是否表示数值(包括整数和小数)。例如,字符串 "+100"、"5e2"、"-123"、"3.1415" 以及 "-1E-16" 都表示数值,但"12e"、"1a3.14"、"1.2.3"、"+-5" 以及 "12e+5.4" 都不是

2. 思路

表示数值的字符串遵循模式 A[.[B]][e|EC] 或者 .B[e|EC],其中 A 为数值的整数部分,B 紧跟着小数点为数值的小数部分,C 紧跟着 'e' 或者 'E' 为数值的指数部分。在消暑利可能没有数值的整数部分。例如,小数 .123 等于 0.123。因此 A 部分不是必需的。如果一个数没有整数部分,那么它的小数部分不能为空。

上述 A 和 C 都是可能以 '+' 或者 '-' 开头的 0~9 的数位串;B 也是 0~9 的数位串,但前面不能有正负号。

以表示数值的字符串 "123.45e+6" 为例,"123" 是它的整数部分 A,"45" 是它的小数部分 B,"+6" 是它的指数部分 C。

判断一个字符串是否符合上述模式,首先应该扫描 0~9 的数位,也就是上面数值中表示整数的 A 部分。如果遇到小数点,则开始扫描数值小数部分 B。如果遇到 'e' 或者 'E',则开始扫描表示数值指数的 C 部分。根据这样的思路,给出参考代码:

public class Main{
public boolean isNumeric(char[] str){
if(str==null)
return false;
// 这个数组就一个元素,存储遍历字符串的下标
int[] index = {0};
// 先遍历字符串的整数部分
boolean numeric = scanInteger(str,index);
// 碰到小数点时,开始遍历 B 部分
if(index[0]<str.length && str[index[0]]=='.'){
index[0]++;
/*
* 这里使用 || 的原因
* 1. 小数可以没有整数,如 .123 等于 0.123
* 2. 小数点后面可以没有数字,如 223. 等于 233.0
* 3. 当然,小数点前面和后面都可以有数字,如 2233.444
*/
numeric = scanUnsignedInteger(str,index) || numeric;
}
// 如果出现 'e' 或者 'E',开始遍历指数部分
if(index[0]<str.length && (str[index[0]]=='e' || str[index[0]]=='E')){
index[0]++;
/*
* 这里使用 && 的原因
* 1. 当 e 或 E 前面没有数字时,整个字符串不能表示数字,如 .e1、e2。
* 2. 当 e 或 E 后面没有整数时,整个字符串不能表示数字,如 23e、23e+5.4
*/
numeric = numeric && scanInteger(str,index);
}
return index[0]==str.length && numeric;
} /*
* 该函数用来扫描字符串中 0~9 的数位(类似于一个无符号整数),可以用来匹配前面数值
* 模式中的 B 部分
*/
private boolean scanInteger(char[] str,int[] index){
if(index[0]<str.length && (str[index[0]]=='+' || str[index[0]]=='-'))
index[0]++;
return scanUnsignedInteger(str,index);
} /*
* 该函数扫描可能以表示正负号的 '+' 或者 '-' 开头的 0~9 的数位(类似于一个有符号整数)
* 可以用来匹配前面数值模式中的 A 和 C 部分。
*/
private boolean scanUnsignedInteger(char[] str,int[] index){
int before = index[0];
while(index[0]<str.length && str[index[0]]>='0' && str[index[0]]<='9')
index[0]++;
return index[0]>before;
}
}

更多《剑指offer》的题目以及源代码可以参考我的 github

剑指offer之字符串是否为数值的更多相关文章

  1. 剑指Offer:字符串排列【38】

    剑指Offer:字符串排列[38] 题目描述 输入一个字符串,按字典序打印出该字符串中字符的所有排列.例如输入字符串abc,则打印出由字符a,b,c所能排列出来的所有字符串abc,acb,bac,bc ...

  2. 剑指 Offer 38. 字符串的排列 + 无重复元素的全排列

    剑指 Offer 38. 字符串的排列 Offer_38 题目描述 解题思路 可以使用递归实现全排列,每次都确定一个数的位置,当所有位置的数都确定后即表示一个排列. 但是考虑到本题需要排除重复的排列, ...

  3. 剑指 Offer 38. 字符串的排列

    剑指 Offer 38. 字符串的排列 输入一个字符串,打印出该字符串中字符的所有排列. 你可以以任意顺序返回这个字符串数组,但里面不能有重复元素. 示例: 输入:s = "abc" ...

  4. 【剑指Offer】字符串的排列 解题报告(Python)

    [剑指Offer]字符串的排列 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https://www.nowcoder.com/ta/coding-interviews 题 ...

  5. 剑指offer 67. 字符串转换为整数(Leetcode 8. String to Integer (atoi))

    题目:剑指offer 67题 需要考虑的情况:空指针.nullptr.空字符串"".正负号.数值溢出.在写代码的时候对这些特殊的输入都定义好合理的输出.可以定义一个全局布尔型变量g ...

  6. 【Java】 剑指offer(38) 字符串的排列

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 输入一个字符串,打印出该字符串中字符的所有排列.例如输入字符串ab ...

  7. 【Java】 剑指offer(50-1) 字符串中第一个只出现一次的字符

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 在字符串中找出第一个只出现一次的字符.如输入"abacc ...

  8. 【剑指offer】字符串的组合

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/mmc_maodun/article/details/26405471 转载请注明出处:http:// ...

  9. Go语言实现:【剑指offer】字符串的排列

    该题目来源于牛客网<剑指offer>专题. 输入一个字符串,按字典序打印出该字符串中字符的所有排列.例如输入字符串abc,则打印出由字符a,b,c所能排列出来的所有字符串abc,acb,b ...

随机推荐

  1. bzoj4459[Jsoi2013]丢番图

    bzoj4459[Jsoi2013]丢番图 题意: 丢番图方程:1/x+1/y=1/n(x,y,n∈N+) ,给定n,求出关于n的丢番图方程有多少组解.n≤10^14. 题解: 通分得yn+xn=xy ...

  2. 利用宝塔和rainloop搭建咱的邮箱

    需要咱准备的东东:一枚域名.服务器需根据情况开放25.110.143.465.993端口.宝塔邮局管理器.rainloop.LNMP或者LAMP. 搭建步骤: 1.安装宝塔邮局管理器: 2.设置宝塔邮 ...

  3. JAVA集合三:几种Set框架

    参考链接: HOW2J.CN HashSet简单讲解 HashSet HashSet与C++STL中Set基本类似,具有的特点便是: 集合中元素不可重复 集合中元素顺序 ≠ 插入顺序 常用方法 功能 ...

  4. 【C++】初次学习C++指针时的一些易混或疑惑的地方

    C++中的指针是一个比较复杂的知识概念,最近我有在学习这一方面的知识,就借此文章记录一下在学习时容易产生的混淆.本人初次发技术类的分享,可能会有纰漏,欢迎诸位指正^_^! 1.*在两种语境下的含义 先 ...

  5. C# WebClient几种常用方法的用法

    1.UploadData方法(Content-Type:application/x-www-form-urlencoded) //创建WebClient 对象            WebClient ...

  6. form表单两种提交方式的不同

      我们在使用<Form>表单的时候,最常用的提交方式就是Get和Post.我们都知道这两种方式最大的差别就是安全性,除此之外,它们还有哪些其他的区别,你知道吗?   在<Form& ...

  7. rpm -ivh vsftpd-3.0.2-22.el7.x86_64.rpm出现error: open of vsftpd-3.0.2-22.el7.x86_64.rpm failed: No such file or directory的解决方法

    情况一: 出现如图问题, 我当时的问题是通过安装rpmbuild工具软件解决的 以前制作rpm时,没有遇到过这个问题,几经搜索也没有解决.后来发现当前的centos没有安装 rpmbuild 工具软件 ...

  8. 【扩展推荐】Intervention/image 图片处理

    Intervention/image 是为 Laravel 定制的图片处理工具, 它提供了一套易于表达的方式来创建.编辑图片. 一.环境要求 二.安装及配置 下载地址:https://packagis ...

  9. MyBatis----resultMap的使用

  10. __getattribute__小例子

    class student(object): def __init__(self,name=None,age=None): self.name = name self.age = age def __ ...