当前位置: 首页 > article >正文

org.apache.commons.lang.math.NumberUtils#isNumber 解释

源码


    /**
     * <p>Checks whether the String a valid Java number.</p>
     *
     * <p>Valid numbers include hexadecimal marked with the <code>0x</code>
     * qualifier, scientific notation and numbers marked with a type
     * qualifier (e.g. 123L).</p>
     *
     * <p><code>Null</code> and empty String will return
     * <code>false</code>.</p>
     *
     * @param str  the <code>String</code> to check
     * @return <code>true</code> if the string is a correctly formatted number
     */
    public static boolean isNumber(String str) {
        if (StringUtils.isEmpty(str)) {
            return false;
        }
        char[] chars = str.toCharArray();
        int sz = chars.length;
        boolean hasExp = false;
        boolean hasDecPoint = false;
        boolean allowSigns = false;
        boolean foundDigit = false;
        // deal with any possible sign up front
        int start = (chars[0] == '-') ? 1 : 0;
        if (sz > start + 1) {
            if (chars[start] == '0' && chars[start + 1] == 'x') {
                int i = start + 2;
                if (i == sz) {
                    return false; // str == "0x"
                }
                // checking hex (it can't be anything else)
                for (; i < chars.length; i++) {
                    if ((chars[i] < '0' || chars[i] > '9')
                        && (chars[i] < 'a' || chars[i] > 'f')
                        && (chars[i] < 'A' || chars[i] > 'F')) {
                        return false;
                    }
                }
                return true;
            }
        }
        sz--; // don't want to loop to the last char, check it afterwords
              // for type qualifiers
        int i = start;
        // loop to the next to last char or to the last char if we need another digit to
        // make a valid number (e.g. chars[0..5] = "1234E")
        while (i < sz || (i < sz + 1 && allowSigns && !foundDigit)) {
            if (chars[i] >= '0' && chars[i] <= '9') {
                foundDigit = true;
                allowSigns = false;

            } else if (chars[i] == '.') {
                if (hasDecPoint || hasExp) {
                    // two decimal points or dec in exponent   
                    return false;
                }
                hasDecPoint = true;
            } else if (chars[i] == 'e' || chars[i] == 'E') {
                // we've already taken care of hex.
                if (hasExp) {
                    // two E's
                    return false;
                }
                if (!foundDigit) {
                    return false;
                }
                hasExp = true;
                allowSigns = true;
            } else if (chars[i] == '+' || chars[i] == '-') {
                if (!allowSigns) {
                    return false;
                }
                allowSigns = false;
                foundDigit = false; // we need a digit after the E
            } else {
                return false;
            }
            i++;
        }
        if (i < chars.length) {
            if (chars[i] >= '0' && chars[i] <= '9') {
                // no type qualifier, OK
                return true;
            }
            if (chars[i] == 'e' || chars[i] == 'E') {
                // can't have an E at the last byte
                return false;
            }
            if (chars[i] == '.') {
                if (hasDecPoint || hasExp) {
                    // two decimal points or dec in exponent
                    return false;
                }
                // single trailing decimal point after non-exponent is ok
                return foundDigit;
            }
            if (!allowSigns
                && (chars[i] == 'd'
                    || chars[i] == 'D'
                    || chars[i] == 'f'
                    || chars[i] == 'F')) {
                return foundDigit;
            }
            if (chars[i] == 'l'
                || chars[i] == 'L') {
                // not allowing L with an exponent
                return foundDigit && !hasExp;
            }
            // last character is illegal
            return false;
        }
        // allowSigns is true iff the val ends in 'E'
        // found digit it to make sure weird stuff like '.' and '1E-' doesn't pass
        return !allowSigns && foundDigit;
    }

NumberUtils#isNumber 解释

org.apache.commons.lang.math.NumberUtils#isNumber 的作用是检查一个字符串是否可以被解析为一个有效的数字。这包括整数、浮点数、科学计数法表示的数以及十六进制数。下面是对代码的逐步解释:

  1. 初始检查:首先,使用 StringUtils.isEmpty(str) 检查输入字符串是否为空或为 null。如果是,方法返回 false

  2. 处理十六进制数:接下来,代码检查字符串是否以 “0x” 开头,这表明它可能是一个十六进制数。如果是,它将遍历字符串的其余部分,确保每个字符都是有效的十六进制数字(0-9, a-f, A-F)。如果所有字符都有效,方法返回 true

  3. 设置循环变量和条件:代码设置了几个变量来跟踪解析过程中的状态,包括是否发现了指数部分 (hasExp)、是否有小数点 (hasDecPoint)、是否允许符号 (allowSigns)、是否找到了数字 (foundDigit)。循环从字符串的开始(或跳过了负号后的第一个字符)开始,直到字符串的倒数第二个字符,或者在需要另一个数字来形成有效数字时直到最后一个字符。

  4. 循环解析字符:在循环中,代码根据当前字符的类型更新状态变量。如果字符是数字,foundDigit 设置为 trueallowSigns 设置为 false。如果字符是小数点,检查是否已经有小数点或指数部分,如果有,则返回 false。如果字符是 ‘e’ 或 ‘E’,检查是否已经有指数部分或之前没有找到数字,如果是,则返回 false;否则,设置 hasExptrue 并允许后续出现符号。如果字符是 ‘+’ 或 ‘-’,检查是否允许符号,如果不允许,则返回 false;否则,重置 allowSignsfoundDigit

  5. 检查最后一个字符:循环结束后,代码检查字符串的最后一个字符。如果是数字,返回 true。如果是 ‘e’ 或 ‘E’,返回 false,因为不能以指数符号结束。如果是小数点,检查是否已经有小数点或指数部分,如果没有且之前找到了数字,则返回 true。如果是类型限定符(‘d’, ‘D’, ‘f’, ‘F’, ‘l’, ‘L’),根据之前的状态返回相应的结果。

  6. 返回结果:最后,如果没有找到不合法的情况,检查 allowSignsfoundDigit 的状态,如果没有挂起的指数符号且至少找到了一个数字,则返回 true;否则,返回 false

总之,这段代码通过一系列的检查和状态跟踪,来判断一个字符串是否可以被解析为一个有效的数字。

NumberUtils.isNumber 示例

下面是一些使用 NumberUtils.isNumber(String str) 方法的示例,以及每个示例的预期返回结果。这些示例展示了不同类型的字符串输入和该方法如何判断它们是否可以被解析为有效的数字。

NumberUtils.isNumber("123"); // true - 正整数
NumberUtils.isNumber("-123"); // true - 负整数
NumberUtils.isNumber("123.45"); // true - 正浮点数
NumberUtils.isNumber("-123.45"); // true - 负浮点数
NumberUtils.isNumber("0.123"); // true - 小于1的浮点数
NumberUtils.isNumber("-0.123"); // true - 小于0的负浮点数
NumberUtils.isNumber("123E4"); // true - 科学计数法
NumberUtils.isNumber("-123E4"); // true - 负的科学计数法
NumberUtils.isNumber("123.45E-6"); // true - 带小数点的科学计数法
NumberUtils.isNumber("0x1A"); // true - 十六进制数
NumberUtils.isNumber("0x1G"); // false - 无效的十六进制数('G' 不是十六进制字符)
NumberUtils.isNumber("123L"); // true - 长整型
NumberUtils.isNumber("NaN"); // false - 不是有效的数字格式
NumberUtils.isNumber("INF"); // false - 不是有效的数字格式
NumberUtils.isNumber(""); // false - 空字符串
NumberUtils.isNumber(null); // false - null 值

请注意,这些示例假设 NumberUtils.isNumber(String str) 方法的实现与之前讨论的代码逻辑一致。实际的返回结果可能会根据具体实现的细节有所不同。

NumberUtils.isCreatable(str) 和 NumberUtils.isNumber(str) 区别

NumberUtils.isCreatable(String str)NumberUtils.isNumber(String str) 都是 Apache Commons Lang 库中的方法,用于判断一个字符串是否可以解析为一个数字。它们之间的主要区别在于它们对可解析数字的定义和容忍度。

  1. NumberUtils.isNumber(String str)(在Apache Commons Lang 3.4及之前的版本中使用):

    • 这个方法用于判断一个字符串是否可以解析为一个有效的Java数字,包括整数、浮点数、和科学计数法表示的数字。
    • 它不接受十六进制格式的数字字符串。
    • 它对于一些边界情况的处理可能不够严格,例如,对于以多个零开头的字符串可能会返回 true
  2. NumberUtils.isCreatable(String str)(在Apache Commons Lang 3.5及之后的版本中引入):

    • 这个方法提供了一个更全面和严格的方式来判断一个字符串是否可以解析为一个数字。
    • 它接受更广泛的数字格式,包括十六进制、八进制和科学计数法。
    • 它更严格地验证字符串格式,以确保解析的数字是有效的。例如,它会检查十六进制数是否只包含有效的十六进制字符。

简而言之,NumberUtils.isCreatable(String str)NumberUtils.isNumber(String str) 的一个改进和扩展版本,提供了更广泛的数字格式支持和更严格的验证。如果你的项目使用的是 Apache Commons Lang 3.5 或更高版本,建议使用 NumberUtils.isCreatable(String str) 来判断字符串是否可以解析为数字。


http://www.kler.cn/a/285168.html

相关文章:

  • 【leetcode练习·二叉树】用「分解问题」思维解题 II
  • 通用项目工程的过程视图概览
  • JavaScript——函数、事件与BOM对象
  • 鸿蒙华为商城APP案例
  • 第七部分:2. STM32之ADC实验--AD多通道(AD采集三路传感器模块实验:光敏传感器、热敏传感器、反射式传感器附赠温湿度传感器教程)
  • 面试题之---解释一下原型和原型链
  • 大语言模型数据增强与模型蒸馏解决方案
  • 【最新华为OD机试E卷】空栈压数(200分)-多语言题解-(Python/C/JavaScript/Java/Cpp)
  • 【测试】——开发模型与测试模型
  • 黑神话 悟空 配置 Mac玩游戏
  • vue3中ref绑定的节点顺序错乱
  • day36
  • 【MySQL 12】事务管理 (带思维导图)
  • leetcode 147.对链表进行插入排序
  • Pr:代理预设
  • [E二叉树] lc110. 平衡二叉树(dfs+自底向上)
  • Java技术栈 —— Spark入门(二)之实时WordCount
  • 基于微信小程序的电动车租赁系统---附源码97332
  • 遇到的BUG及解决方法
  • 【读书笔记-《30天自制操作系统》-12】Day13
  • 监控平台之上报(未完成)
  • Python算法工程师面试整理-Python 编程技巧
  • 使用Ansible stat模块检查目录是否存在
  • 【Docker】Dockerfile实列-Nginx镜像构建
  • 类与ES6类之间的继承
  • 叶斯神经网络(BNN)在训练过程中损失函数不收敛或跳动剧烈可能是由多种因素