剑指offer20题表示数值的字符串:这题实在是太优雅了

时间:2022-11-04 19:06:18

题目来源:https://leetcode.cn/problems/biao-shi-shu-zhi-de-zi-fu-chuan-lcof/

这个题目有官方解释的有限状态自动机,也有用正则肝的,甚至有暴躁老哥打算直接把这题撕了的,但是,还得是评论区里人才多,总有大佬能肝出来清晰的题解,这波记录一下,食用这个题目的过程。
剑指offer20题表示数值的字符串:这题实在是太优雅了
大佬讲解链接:https://www.bilibili.com/video/BV1KP411L7VH/?p=20&vd_source=a3808e6f1547cd8a5765ed7361e92529

一、憨憨初解

1、思路

看完题目感觉没什么难的,字符串处理题么应该是,从字符串开始到最后遍历就完事儿,用cur记录遍历的长度,如果最后能够按照题目给的规则遍历完,那么cur就等于字符串的长度就返回true,否则false

2、代码

class Solution {
public:
    bool isNumber(string s) {
        if (s.empty() || s.size() == 0) return false;
        int cur=0;
        while(s[cur] == ' ')  cur++;//去调首部空格
        //处理一个整数或者小数
        if(s[cur] == '+' || s[cur] == '-') cur++;
        while(s[cur] >= '0' && s[cur] <= '9')    cur++;
        if(s[cur] == '.') cur++;
        while(s[cur] >= '0' && s[cur] <= '9')    cur++;

        //处理e和E以及一个整数
        if(s[cur]=='e' || s[cur]=='E') cur++;
        if(s[cur] == '+' || s[cur] == '-') cur++;
        while(s[cur] >= '0' && s[cur] <= '9')    cur++;

        while(s[cur] == ' ')  cur++;//处理尾部空格
        return cur==s.size();
    }
};

3、战绩

剑指offer20题表示数值的字符串:这题实在是太优雅了

4、反思

存在一些情况没有考虑,憨憨我直接一波下去了,看题解看的有点懵,然后看大佬视频题解后,理解了思路,自己用C++复现了一波;

二、看懂再解

1、思路

首先判断为空字符串的返回false,然后用trim去除段首段尾空格;
定义三个标准布尔量,分别表示数字和点还要e或者E是否出现过,从开始到结尾进行遍历,如果出现了不符合标准的情况就返回false。

这个不符合标准的情况规律是大佬以身探出来的,感恩一下前人,自己很难严谨的考虑到这些东西。

1、小数点出现的时候,前面不能出现过小数点或者是出现过e;

  • 解释:'.'只能在底数上,不能在指数上,且只能出现一次,'.'两边任一边有数字均算一个完整的数字,但单独一个'.'不行。

2、e或者E出现的时候,前面不能出现过eE,并且必须出现过一个数字;

  • 解释:e、E用来划分底数与指数,只能出现一次,前面为科学计数法的底数,后面为指数;

3、+-符号,只能出现在第一位或者紧接e后面;

  • 解释:'+'、'-'只能作为正负号出现在底数和指数的前面,不能出现在两个数字中间;

特别需要注意的是为了避免123e这种请求,出现e之后就标志is_num为false,最后返回is_num即可!
其实不用看解释,理解即可,记好这三条规律和一个特别的情况就能快速的编写出解题代码!

2、代码

复现的C++版:

class Solution {
public:
    static void trim(string &s){
        if( !s.empty() ){
            s.erase(0,s.find_first_not_of(" "));
            s.erase(s.find_last_not_of(" ") + 1);
        }
    }
    bool isNumber(string s) {
        if (s.empty() || s.size() == 0) return false;
        trim(s);
        bool is_num = false;//判断数字是否出现过
        bool is_dot = false;//判断.是否出现过
        bool is_eorE = false;//判断e或者E是否出现过
        int i = 0,size = s.size();
        for (; i < size; i++) {
            //判定为数字,则标记is_num
            if (s[i] >= '0' && s[i] <= '9') {   
                is_num = true;
            //判定为.  需要前面没出现过.并且没出现过e
            } else if (s[i] == '.' && !is_dot && !is_eorE) {  
                is_dot = true;       
            //判定为e或者E,需要前面没出现过e,并且出现过数字
            } else if ((s[i]=='e' || s[i]=='E') && !is_eorE && is_num) {
                is_eorE = true;
                is_num = false;//为了避免123e这种请求,出现e之后就标志为false      
            //判定为+-符号,只能出现在第一位或者紧接e后面
            } else if (s[i] == '-' || s[i] == '+'){
                if(i!=0 && s[i-1] != 'e' && s[i-1] != 'E'){
                    return false;
                }
            } else {
                return false;
            }
        }
        return is_num;
    }
};

大佬的Java版:

class Solution {
    public boolean isNumber(String s) {
        //有限状态机
        // 2.小数点 3.E/e 4. 数字字符 5. -+
        if(s == null || s.length() <= 0){
            return false;
        }
        char[] res = s.trim().toCharArray();
        if(res.length <= 0) return false;

        int n = res.length;

        boolean is_dot = false;
        boolean is_e_or_E = false;
        boolean is_num = false;
        for(int i = 0; i < n; i++){
            if(res[i] >= '0' && res[i] <= '9'){
                is_num = true;
            } else if(res[i] == '.'){
                //-+ 8.  8.8  .8
                // 前面:不能有重复的小数点,也不能出现 e/E
                if(is_dot || is_e_or_E){
                    return false;
                }
                is_dot = true;
            } else if(res[i] == 'e' || res[i] == 'E'){
                // 前面必须要有一个数字 || 前面不能出现重复的 e/E
                if(is_e_or_E || !is_num){
                    return false;
                }
                is_e_or_E = true;
                is_num =false;//11E+ 11E
            } else if(res[i] == '-' || res[i] == '+'){
                if(i!=0 && res[i-1] != 'e' && res[i-1] != 'E'){
                    return false;
                }
            } else {
                return false;
            }
        }

        return is_num;
    }
}

3、C++版战绩

剑指offer20题表示数值的字符串:这题实在是太优雅了

总结

高情商:考察编程代码严谨能力!
低情商:不太想要你。

开玩笑的蛤,不要太当真,不管什么问题,都要能够具备解决它的能力,这题虽然比较难受,但是还是学到了一点东西的。
发现了一个宝藏博主:https://www.cnblogs.com/Shirlies/p/4666744.html
大佬题解页面:https://www.playoffer.cn/84.html