gpt4 book ai didi

java - 在java中解析带有尾随垃圾的 float

转载 作者:行者123 更新时间:2023-12-02 17:39:42 26 4
gpt4 key购买 nike

背景:我正在尝试增量解析“cos(1.2)”等表达式。现在,到实际问题(注意:实际问题主要在下一段中;其余的都是关于似乎几乎有效的解决方案的漫谈):

假设我有一个 Java 中的字符串,它可能以 float 开头,然后后面有一些更多的“东西”。例如,我可能有 52hi(以“52”开头,以“hi”结尾)或 -1.2e1e9(以“-1.2e1”开头,也称为“负十二”,以“e9”结尾) )。我想将这个数字解析为 double 。

使用 Double.parseDouble 很诱人,但此方法期望字符串作为一个整体是有效的数字,如果不是,则会抛出异常。显而易见的事情是编写一个正则表达式将数字与其他内容分开,然后使用 parseDouble。

如果我正在解析整数,这不会太糟糕,比如 -?[0-9]+。 (即便如此,很容易忘记边缘情况,现在您的用户无法输入 +9 与 -9 对称。因此前面的正则表达式应该是 [-+]?[0-9]+.) 但对于 float 来说就很复杂了;也许是这样的(忽略“.”在大多数正则表达式方言中默认不按字面意思理解的事实):

[-+]?[0-9]*.?[0-9]*(e[-+]?[0-9]+)?

只不过我们刚刚说过空字符串是有效数字。 “.e2”也是如此。所以可能有点复杂。或者也许我可以有一个像上面这样的“草率”正则表达式,只要它不禁止任何实际数字,它就允许一些非数字。但在某些时候,我开始对自己思考“这不是 parseDouble 的工作吗?”。它正在完成找出字符串中数字结束位置和其他内容开始位置所需的大部分工作,因为否则它将无法抛出异常。为什么我也必须这么做?

所以我开始寻找 Java 标准库中是否还有其他可以提供帮助的内容。我通常选择的工具是 java.util.Scanner,它有一个很好的 nextDouble() 方法。但 Scanner 工作于“ token ”,因此 nextDouble 的真正意思是“获取下一个 token 并尝试将其解析为 double ”。标记由分隔符分隔,我的默认分隔符是空格。因此,扫描仪在处理“52 hi”时不会有任何问题,但无法处理“52hi”。理论上,分隔符可以是我选择的任何正则表达式,因此我所要做的就是编写一个正则表达式,当它匹配时,表示数字的结尾。但这似乎比直接编写正则表达式更难。

当我发现 java.text.DecimalFormat 时,我正要放弃希望,它明确表示“我将尽我所能进行解析,并且我会告诉你我已经走了多远,以便你可以继续做其他事情从那时起”。但它似乎主要是为了格式化供人类使用的东西而设计的,并且可能解析机器编写的东西,但不是解析人类编写的东西,并且它以一堆小方式出现。例如,它“支持”像“1.2e1”这样的科学记数法,但是如果您使用它,它会坚持数字必须采用科学记数法,如果您输入“12”,它会导致解析失败。人们可以尝试通过检查失败的位置并将之前的内容解析为数字来解决此问题,但这很容易出错,甚至比只为 float 编写正则表达式更烦人。

同时,在 C 中,这只是 sscanf("%f"),而在 C++ 中,您可以使用字符串流来完成基本相同的操作。 Java 中真的没有类似的东西吗?

最佳答案

documentation for Double.valueOf(String) 实际上包含一个正则表达式,您可以使用它来检查字符串是否为 double

这里,没有注释:

final String Digits     = "(\\p{Digit}+)";
final String HexDigits = "(\\p{XDigit}+)";
final String Exp = "[eE][+-]?"+Digits;
final String fpRegex =
("[\\x00-\\x20]*"+
"[+-]?(" +
"NaN|"+
"Infinity|" +
"((("+Digits+"(\\.)?("+Digits+"?)("+Exp+")?)|"+
"(\\.("+Digits+")("+Exp+")?)|"+
"((" +
"(0[xX]" + HexDigits + "(\\.)?)|" +
"(0[xX]" + HexDigits + "?(\\.)" + HexDigits + ")" +
")[pP][+-]?" + Digits + "))" +
"[fFdD]?))" +
"[\\x00-\\x20]*");

您可以像这样使用它:

Matcher m = Pattern.compile(fpRegex).matcher(input);
if (m.find()) {
String doublePartOnly = m.group();
}

通过一些基本测试,我发现正则表达式贪婪的,因此它将匹配1.2e1hello中的1.2e1,而不是只是1.2

关于java - 在java中解析带有尾随垃圾的 float ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60441195/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com