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

server/2024/10/18 12:23:08/

源码

java">/*** <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 frontint 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 qualifiersint 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'sreturn 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, OKreturn true;}if (chars[i] == 'e' || chars[i] == 'E') {// can't have an E at the last bytereturn false;}if (chars[i] == '.') {if (hasDecPoint || hasExp) {// two decimal points or dec in exponentreturn false;}// single trailing decimal point after non-exponent is okreturn 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 exponentreturn foundDigit && !hasExp;}// last character is illegalreturn false;}// allowSigns is true iff the val ends in 'E'// found digit it to make sure weird stuff like '.' and '1E-' doesn't passreturn !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) 方法的示例,以及每个示例的预期返回结果。这些示例展示了不同类型的字符串输入和该方法如何判断它们是否可以被解析为有效的数字。

java">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.ppmy.cn/server/109674.html

相关文章

Python优化算法19——混沌精英黏菌优化算法(CESMA)

科研里面优化算法都用的多&#xff0c;尤其是各种动物园里面的智能仿生优化算法&#xff0c;但是目前都是MATLAB的代码多&#xff0c;python几乎没有什么包&#xff0c;这次把优化算法系列的代码都从底层手写开始。 需要看以前的优化算法文章可以参考&#xff1a;Python优化算…

CSS3多行多栏布局

当前布局由6个等宽行组成&#xff0c;其中第四行有三栏&#xff0c;第五行有四栏。 重点第四行设置&#xff1a; 代码&#xff1a; <!DOCTYPE html> <html><head><meta charset"utf-8"><title></title><style>img {hei…

【ES常用查询操作】

在 Elasticsearch (ES) 中&#xff0c;多条件检索可以通过组合多个查询条件来实现。Elasticsearch 提供了多种查询类型和组合方式&#xff0c;常见的有 bool 查询、must、should、must_not 和 filter 等。以下是一些常见的多条件检索示例。 1. 使用 bool 查询 bool 查询允许你…

【设计模式】简单工厂模式

❓首先什么是设计模式&#xff1f; &#x1f635;相信刚上大学的你和我一样&#xff0c;在学习这门课的时候根本不了解这些设计原则和模式有什么用处&#xff0c;反而不如隔壁的C更有意思&#xff0c;至少还能弹出一个小黑框&#xff0c;给我个hello world。 ✨ 如何你和我一样…

MySQL:简述对事务的认识

浅谈对Spring事务的认识&#xff1a;https://xiaoer.blog.csdn.net/article/details/80849971 一、事务的特性 事务是数据库永恒不变的话题&#xff0c; ACID&#xff1a;原子性&#xff0c;一致性&#xff0c;隔离性&#xff0c;持久性。 &#xff08;1&#xff09;原子性&am…

Python中排序算法之冒泡排序

排序算法是将给定的数列中的数进行升序&#xff08;从小到大&#xff09;或者降序&#xff08;从大到小&#xff09;排列。冒泡排序是排序算法的一种。 1 冒泡排序的原理 1.1 基本思想 冒泡排序是将数据中较大或者较小的数据依次向右推移的一种排序技术。它的基本思想是比较…

基于django的失物招领系统的设计与实现/ 基于Python的失物招领系统的设计与实现/失物招领管理系统

失物招领系统的设计与实现 摘要:伴随着我国全面推动信息化的趋势&#xff0c;我国的很多行业都在朝着互联网的方向进发。结合计算机技术的失物招领系统能够很好地解决传统失物招领存在的问题&#xff0c;能够提高管理员管理的效率&#xff0c;改善服务质量。优秀的失物招领系统…

低代码技术助力移动端开发:简化开发流程,实现快速创新

在移动互联网快速发展的今天&#xff0c;企业和开发者面临着越来越高的需求&#xff0c;要求开发高质量、功能强大的移动应用&#xff0c;以满足用户的期待和市场的变化。然而&#xff0c;传统的移动端开发流程通常复杂且耗时&#xff0c;需要投入大量的资源和开发人员。为了应…