表示数值的字符串请实现一个函数用来判断字符串是否表示数值(包括整数和小数)。例如,字符串"+100"、"5e2"、"-123"、"3.1416"、"-1E-16"、"0123"都表示数值,但"12e"、"1a3.14"、"1.2.3"、"+-5"及"12e+5.4"都不是。
DFA有限状态自动机
时间复杂度 O(N) : 其中 N 为字符串 s 的长度,判断需遍历字符串,每轮状态转移的使用 O(1) 时间。 空间复杂度 O(1): states 和 p 使用常数大小的额外空间。
class Solution {
public boolean isNumber(String s) {
Map[] states = {
new HashMap<>() {{ put(' ', 0); put('s', 1); put('d', 2); put('.', 4); }}, // 0.
new HashMap<>() {{ put('d', 2); put('.', 4); }}, // 1.
new HashMap<>() {{ put('d', 2); put('.', 3); put('e', 5); put(' ', 8); }}, // 2.
new HashMap<>() {{ put('d', 3); put('e', 5); put(' ', 8); }}, // 3.
new HashMap<>() {{ put('d', 3); }}, // 4.
new HashMap<>() {{ put('s', 6); put('d', 7); }}, // 5.
new HashMap<>() {{ put('d', 7); }}, // 6.
new HashMap<>() {{ put('d', 7); put(' ', 8); }}, // 7.
new HashMap<>() {{ put(' ', 8); }} // 8.
};
int p = 0;
char t;
for(char c : s.i()) {
# 获取当前的状态
if(c >= '0' && c <= '9') t = 'd'; #数字太多一次判断不完,用d表示
else if(c == '+' || c == '-') t = 's'; #正负号用+-
else if(c == 'e' || c == 'E') t = 'e'; #大小写统一
else if(c == '.' || c == ' ') t = c; #点和空格可以用自身表示
else t = '?'; #非法字符
if(!states[p].containsKey(t)) return false; #状态非法
#状态转移到下一个位置
p = (int)states[p].get(t);
}
return p == 2 || p == 3 || p == 7 || p == 8;#终态在2378中就接受
}
}