跳至主要內容

20. 表示数值的字符串


20. 表示数值的字符串

🟠   🔖  字符串  🔗 力扣open in new window

题目

有效数字 (按顺序)可以分成以下几个部分:

  1. 若干空格
  2. 一个 小数 或者 整数
  3. (可选)一个 'e''E' ,后面跟着一个 整数
  4. 若干空格

小数 (按顺序)可以分成以下几个部分:

  1. (可选)一个符号字符('+''-'
  2. 下述格式之一:
    1. 至少一位数字,后面跟着一个点 '.'
    2. 至少一位数字,后面跟着一个点 '.' ,后面再跟着至少一位数字
    3. 一个点 '.' ,后面跟着至少一位数字

整数 (按顺序)可以分成以下几个部分:

  1. (可选)一个符号字符('+''-'
  2. 至少一位数字

部分有效数字列举如下:["2", "0089", "-0.1", "+3.14", "4.", "-.9", "2e10", "-90E3", "3e+7", "+6e-1", "53.5e93", "-123.456e789"]

部分无效数字列举如下:["abc", "1a", "1e", "e3", "99e2.5", "--6", "-+3", "95a54e53"]

给你一个字符串 s ,如果 s 是一个 有效数字 ,请返回 true

示例 1:

输入: s = "0"

输出: true

示例 2:

输入: s = "e"

输出: false

示例 3:

输入: s = "."

输出: false

提示:

  • 1 <= s.length <= 20
  • s 仅含英文字母(大写和小写),数字(0-9),加号 '+' ,减号 '-' ,空格 ' ' 或者点 '.'

解题思路

思路一 正则表达式

为了判断一个字符串是否是有效数字,我们可以使用正则表达式来匹配其格式。根据题目的描述,我们可以构建一个正则表达式来验证字符串的有效性。

正则表达式构建

根据题目要求,有效数字的格式可以用以下正则表达式表示:

  1. 可选的空格:使用 ^\s*\s*$
  2. 有效数字部分
    • 可以是一个整数或小数,后面可以跟可选的科学计数法部分('e' 或 'E' 后跟整数)。
    • 小数和整数的详细结构在描述中已给出。

正则表达式示例

以下是构建的正则表达式:

^\s*[+-]?((\d+(\.\d*)?)|(\.\d+))([eE][+-]?\d+)?\s*$

  • ^\s*\s*$ 用于匹配前后的空格。
  • [+-]? 表示可选的符号。
  • (\d+(\.\d*)?)|(\.\d+) 用于匹配小数和整数:
    • \d+(\.\d*)? 匹配整数和可选的小数部分。
    • \.\d+ 匹配以点开始的数(如 .5)。
  • ([eE][+-]?\d+)? 匹配科学计数法部分。

复杂度分析

  • 时间复杂度O(n),其中 n 是字符串的长度,正则表达式的匹配过程会遍历整个字符串一次,因此时间复杂度为线性。
  • 空间复杂度O(1),不使用额外的空间,正则表达式的匹配结果通常在常量空间内完成,不会额外使用与输入规模相关的空间。

思路二 手动解析字符串

  1. 去除空格:使用 trim() 函数去掉字符串两端的空格。

  2. 标志位:使用 hasNumhasDothasExp 标志数字、小数点和指数的出现。

  3. 遍历字符串

    • 允许字符串以 '+''-' 开头。
    • 逐字符遍历字符串,判断每个字符是否是数字、小数点或科学计数法的符号。
  4. 检查有效性

    • 确保小数点和指数符号只出现一次。
    • 确保数字部分在小数点前后都有数字(如适用)。
    • 确保指数符号后有数字。
  5. 返回结果:最后根据 hasNum 判断是否存在有效的数字部分。

复杂度分析

  • 时间复杂度O(n),其中 n 是字符串的长度,需要遍历整个字符串一次,因此时间复杂度为线性。
  • 空间复杂度O(1),使用常量空间来存储标志位,不会根据输入规模增加空间使用。

代码

正则表达式
/**
 * @param {string} s
 * @return {boolean}
 */
var validNumber = function (s) {
	const regex = /^\s*[+-]?((\d+(\.\d*)?)|(\.\d+))([eE][+-]?\d+)?\s*$/;
	return regex.test(s);
};