LeetCode之自动机

本文探讨了将字符串转换为整数的常见问题,通过ASCII字符处理和使用确定有限自动机(DFA)的技巧,实现了一个简洁高效的整数解析算法。了解如何利用DFA优化字符串到整数的转换过程,避免冗长判断,提高代码效率。
摘要由CSDN通过智能技术生成

字符串转换整数

请你来实现一个 atoi 函数,使其能将字符串转换成整数。
首先,该函数会根据需要丢弃无用的开头空格字符,直到寻找到第一个非空格的字符为止。接下来的转化规则如下:

如果第一个非空字符为正或者负号时,则将该符号与之后面尽可能多的连续数字字符组合起来,形成一个有符号整数。
假如第一个非空字符是数字,则直接将其与之后连续的数字字符组合起来,形成一个整数。
该字符串在有效的整数部分之后也可能会存在多余的字符,那么这些字符可以被忽略,它们对函数不应该造成影响。
注意:假如该字符串中的第一个非空格字符不是一个有效整数字符、字符串为空或字符串仅包含空白字符时,则你的函数不需要进行转换,即无法进行有效转换。

在任何情况下,若函数不能进行有效的转换时,请返回 0 。

提示:

本题中的空白字符只包括空格字符 ’ ’ 。
假设我们的环境只能存储 32 位大小的有符号整数,那么其数值范围为 [−231, 231 − 1]。如果数值超过这个范围,请返回 INT_MAX (231 − 1) 或 INT_MIN (−231) 。

在只看题目之后,觉得这只是一道很简单的题,然后点进来开描述,发现需要考虑的东西也不少,直接写的话代码肯定会有点臃肿,看起来呆呆的,但是我还是试了一下,于是就有了下面这样的代码

            if (ch == '-') {
                isNegative = -1;
            }
            // 碰到数字,则判断下一个字符是否为数字
            if (Character.isDigit(ch)) {
                int pop = (int)(ch - '0') * isNegative;
                // 判断是否为最后一个数
                if (i == length - 1) {
                    // 判断是否越界
                    if (ans > Integer.MAX_VALUE / 10 || (ans == Integer.MAX_VALUE / 10 && pop > 7)) {
                        return Integer.MAX_VALUE;
                    }
                    if (ans < Integer.MIN_VALUE / 10 || (ans == Integer.MIN_VALUE / 10 && pop < -8)) {
                        return Integer.MIN_VALUE;
                    }
                    // 进行计算
                    ans = ans * 10 + pop;
                } else {
                    // 获取下一个字符
                    char nextChar = str.charAt(i + 1);

                    // 下一个字符不为数字,则计算完,跳出循环 
                    if (!Character.isDigit(nextChar)) {
                        // 判断是否越界
                        if (ans > Integer.MAX_VALUE / 10 || (ans == Integer.MAX_VALUE / 10 && pop > 7)) {
                            return Integer.MAX_VALUE;
                        }
                        if (ans < Integer.MIN_VALUE / 10 || (ans == Integer.MIN_VALUE / 10 && pop < -8)) {
                            return Integer.MIN_VALUE;
                        }
                        ans = ans * 10 + pop;
                        break;
                    } else {
                        // 判断是否越界
                        if (ans > Integer.MAX_VALUE / 10 || (ans == Integer.MAX_VALUE / 10 && pop > 7)) {
                            return Integer.MAX_VALUE;
                        }
                        if (ans < Integer.MIN_VALUE / 10 || (ans == Integer.MIN_VALUE / 10 && pop < -8)) {
                            return Integer.MIN_VALUE;
                        }
                        //下一个数字仍然为数字,则正常计算
                        ans = ans * 10 + pop;
                       
                    }
                }

            }

各种判断,加标记,感觉很呆,看了官方的答案,提到用自动机解决,于是我就去学了一下
DFA全称为:Deterministic Finite Automaton,即确定有穷自动机。其特征为:有一个有限状态集合和一些从一个状态通向另一个状态的边,每条边上标记有一个符号,其中一个状态是初态,某些状态是终态。但不同于不确定的有限自动机,DFA中不会有从同一状态出发的两条边标志有相同的符号。
举个例子说明:
一个仅有ab的字符串,要求b需要成对出现,否则不合法。就是(a|bb)*正则的匹配。我们可以用dfa来做这个题

我们可以通过要求生成这样一个自动机:字符串一共有3种状态,分别是没有b的状态或者b合法的状态,“a”,只有一个b的临时状态“ab”,b不匹配的“aba”状态。
没有输入的时候处于状态1,当输入一个a的时候还是处于状态1。
当输入一个b的时候处于状态2。变成“xxxxab”
当状态2再输入一个b,这是变成“abb”合法,又回到状态1.
当状态2再输入一个a,这时变成了“aba”不合法状态,成为状态3
状态3无论输入什么都是不合法的,都是状态3。

类似的方法来做这题

class Solution {
    public int myAtoi(String str) {
        Automaton automaton = new Automaton();
        int length = str.length();
        for (int i = 0; i < length; ++i) {
            automaton.get(str.charAt(i));
        }
        return (int) (automaton.sign * automaton.ans);
    }
}

class Automaton {
    public int sign = 1;
    public long ans = 0;
    private String state = "start";
    private Map<String, String[]> table = new HashMap<String, String[]>() {{
        put("start", new String[]{"start", "signed", "in_number", "end"});
        put("signed", new String[]{"end", "end", "in_number", "end"});
        put("in_number", new String[]{"end", "end", "in_number", "end"});
        put("end", new String[]{"end", "end", "end", "end"});
    }};

    public void get(char c) {
        state = table.get(state)[get_col(c)];
        if ("in_number".equals(state)) {
            ans = ans * 10 + c - '0';
            ans = sign == 1 ? Math.min(ans, (long) Integer.MAX_VALUE) : Math.min(ans, -(long) Integer.MIN_VALUE);
        } else if ("signed".equals(state)) {
            sign = c == '+' ? 1 : -1;
        }
    }

    private int get_col(char c) {
        if (c == ' ') {
            return 0;
        }
        if (c == '+' || c == '-') {
            return 1;
        }
        if (Character.isDigit(c)) {
            return 2;
        }
        return 3;
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值