如何检查字符串是否可解析为double?

Lou*_*hys 66 java string floating-point parsing

是否有一种本机方式(最好不要实现自己的方法)来检查字符串是否可解析Double.parseDouble()

blu*_*l2k 56

像往常一样,Apache以Apache Commons-Lang的形式 提供了很好的答案Apache

处理空值,不需要Apache Commons-Lang/ NumberUtils.isCreatable(String)阻止.

  • 它为float/double数字返回false. (8认同)
  • 不.您可以在此处找到Apache Commons库.http://commons.apache.org/强烈建议您使用它们,而不是随时编写自定义代码. (5认同)
  • `NumberUtils.isNumber(String)` 已被弃用,您现在应该使用 `NumberUtils.isCreatable(String)` 代替。 (2认同)

jdc*_*589 53

您始终可以在try catch块中包装Double.parseDouble().

try
{
  Double.parseDouble(number);
}
catch(NumberFormatException e)
{
  //not a double
}
Run Code Online (Sandbox Code Playgroud)

  • 这将是我首选的方法,因为您保证获得与解析完全相同的行为.使用自定义正则表达式可以轻松忽略边缘情况. (7认同)
  • @ giant91 - 只修剪数字:`number.trim()`. (3认同)
  • 这是好的,如果你键入一个数字后跟一个空格,它不会捕获它. (2认同)
  • @giant91 所以?该字符串是可解析的,这是实际问题。 (2认同)
  • 我比接受的答案更喜欢这个。这保证在 Java 的所有未来版本中都有效,而使用 RegEx 试图变得聪明并预测解析器的行为。无论如何,正则表达式都很慢,这不像试图找出合适的字符串会比简单地尝试解析它并通过异常捕获过程节省很多时间。 (2认同)

Joh*_*ter 45

常见的方法是使用正则表达式检查它,就像Double.valueOf(String)文档中也提到的那样.

在那里(或包含在下面)提供的正则表达式应该涵盖所有有效的浮点情况,所以你不需要摆弄它,因为你最终会错过一些更精细的点.

如果你不想这样做,try catch仍然是一个选择.

JavaDoc建议的正则表达式如下:

final String Digits     = "(\\p{Digit}+)";
final String HexDigits  = "(\\p{XDigit}+)";
// an exponent is 'e' or 'E' followed by an optionally 
// signed decimal integer.
final String Exp        = "[eE][+-]?"+Digits;
final String fpRegex    =
    ("[\\x00-\\x20]*"+ // Optional leading "whitespace"
    "[+-]?(" +         // Optional sign character
    "NaN|" +           // "NaN" string
    "Infinity|" +      // "Infinity" string

    // A decimal floating-point string representing a finite positive
    // number without a leading sign has at most five basic pieces:
    // Digits . Digits ExponentPart FloatTypeSuffix
    // 
    // Since this method allows integer-only strings as input
    // in addition to strings of floating-point literals, the
    // two sub-patterns below are simplifications of the grammar
    // productions from the Java Language Specification, 2nd 
    // edition, section 3.10.2.

    // Digits ._opt Digits_opt ExponentPart_opt FloatTypeSuffix_opt
    "((("+Digits+"(\\.)?("+Digits+"?)("+Exp+")?)|"+

    // . Digits ExponentPart_opt FloatTypeSuffix_opt
    "(\\.("+Digits+")("+Exp+")?)|"+

    // Hexadecimal strings
    "((" +
    // 0[xX] HexDigits ._opt BinaryExponent FloatTypeSuffix_opt
    "(0[xX]" + HexDigits + "(\\.)?)|" +

    // 0[xX] HexDigits_opt . HexDigits BinaryExponent FloatTypeSuffix_opt
    "(0[xX]" + HexDigits + "?(\\.)" + HexDigits + ")" +

    ")[pP][+-]?" + Digits + "))" +
    "[fFdD]?))" +
    "[\\x00-\\x20]*");// Optional trailing "whitespace"

if (Pattern.matches(fpRegex, myString)){
    Double.valueOf(myString); // Will not throw NumberFormatException
} else {
    // Perform suitable alternative action
}
Run Code Online (Sandbox Code Playgroud)

  • @Louis Rhys - 答案中的文档链接中有一个代码示例. (2认同)
  • Java API 开发人员采用的方法是愚蠢的:他们为我们提供了一大块代码,每个感兴趣的人都必须将其复制/粘贴到实用方法中。为什么不在“Double”类中提供该方法本身呢?:捂脸(没完没了) (2认同)

Coo*_*ans 9

像下面这样的东西就足够了: -

String decimalPattern = "([0-9]*)\\.([0-9]*)";  
String number="20.00";  
boolean match = Pattern.matches(decimalPattern, number);
System.out.println(match); //if true then decimal else not  
Run Code Online (Sandbox Code Playgroud)

  • 实际上它比那更复杂.见johannes wachter的回答 (2认同)
  • 我觉得这个答案非常好。它比其他库灵活得多,并且不依赖于第三方库。 (2认同)
  • 迟到了,但这也将点匹配为有效的双精度数,但它不会将“20”匹配为有效的,而实际上它是有效的。 (2认同)

ruh*_*ong 8

谷歌的番石榴库提供了一个很好的辅助方法来做到这一点:Doubles.tryParse(String).您可以使用它,Double.parseDoublenull如果字符串不解析为double,则返回而不是抛出异常.


Zac*_*h-M 7

所有答案都可以,取决于你想要的学术水平.如果您希望准确地遵循Java规范,请使用以下命令:

private static final Pattern DOUBLE_PATTERN = Pattern.compile(
    "[\\x00-\\x20]*[+-]?(NaN|Infinity|((((\\p{Digit}+)(\\.)?((\\p{Digit}+)?)" +
    "([eE][+-]?(\\p{Digit}+))?)|(\\.((\\p{Digit}+))([eE][+-]?(\\p{Digit}+))?)|" +
    "(((0[xX](\\p{XDigit}+)(\\.)?)|(0[xX](\\p{XDigit}+)?(\\.)(\\p{XDigit}+)))" +
    "[pP][+-]?(\\p{Digit}+)))[fFdD]?))[\\x00-\\x20]*");

public static boolean isFloat(String s)
{
    return DOUBLE_PATTERN.matcher(s).matches();
}
Run Code Online (Sandbox Code Playgroud)

此代码基于Double的JavaDocs .