Python 如何从字符串中提取浮点数

Python 如何从字符串中提取浮点数,python,regex,floating-point,data-extraction,Python,Regex,Floating Point,Data Extraction,我有许多类似于当前级别13.4 db的字符串。我只想提取浮点数。我说的是浮动,而不是小数,因为它有时是完整的。正则表达式能做到这一点吗?或者有更好的方法吗?如果您的浮点值总是用十进制表示,比如 >>> import re >>> re.findall("\d+\.\d+", "Current Level: 13.4 db.") ['13.4'] 也许就够了 更可靠的版本是: >>> re.findall(r"[-+]?\d*\.\d+|\d

我有许多类似于当前级别13.4 db的字符串。我只想提取浮点数。我说的是浮动,而不是小数,因为它有时是完整的。正则表达式能做到这一点吗?或者有更好的方法吗?

如果您的浮点值总是用十进制表示,比如

>>> import re
>>> re.findall("\d+\.\d+", "Current Level: 13.4 db.")
['13.4']
也许就够了

更可靠的版本是:

>>> re.findall(r"[-+]?\d*\.\d+|\d+", "Current Level: -13.2 db or 14.2 or 3")
['-13.2', '14.2', '3']
如果要验证用户输入,也可以通过直接单步执行来检查浮点:

user_input = "Current Level: 1e100 db"
for token in user_input.split():
    try:
        # if this succeeds, you have your (first) float
        print float(token), "is a float"
    except ValueError:
        print token, "is something else"

# => Would print ...
#
# Current is something else
# Level: is something else
# 1e+100 is a float
# db is something else

另一种更具可读性的方法是简单类型转换。我添加了一个替换函数,以涵盖人们可能输入欧洲小数的情况:

>>> for possibility in "Current Level: -13.2 db or 14,2 or 3".split():
...     try:
...         str(float(possibility.replace(',', '.')))
...     except ValueError:
...         pass
'-13.2'
'14.2'
'3.0'

然而,这也有缺点。如果有人输入“1000”,这将转换为1。此外,它还假设人们将在单词之间输入空格。其他语言的情况并非如此,例如汉语。

您可能想尝试这样的方法,它涵盖了所有的基础,包括不依赖数字后面的空格:

>>> import re
>>> numeric_const_pattern = r"""
...     [-+]? # optional sign
...     (?:
...         (?: \d* \. \d+ ) # .1 .12 .123 etc 9.1 etc 98.1 etc
...         |
...         (?: \d+ \.? ) # 1. 12. 123. etc 1 12 123 etc
...     )
...     # followed by optional exponent part if desired
...     (?: [Ee] [+-]? \d+ ) ?
...     """
>>> rx = re.compile(numeric_const_pattern, re.VERBOSE)
>>> rx.findall(".1 .12 9.1 98.1 1. 12. 1 12")
['.1', '.12', '9.1', '98.1', '1.', '12.', '1', '12']
>>> rx.findall("-1 +1 2e9 +2E+09 -2e-9")
['-1', '+1', '2e9', '+2E+09', '-2e-9']
>>> rx.findall("current level: -2.03e+99db")
['-2.03e+99']
>>>
要轻松粘贴副本,请执行以下操作:

numeric_const_pattern = '[-+]? (?: (?: \d* \. \d+ ) | (?: \d+ \.? ) )(?: [Ee] [+-]? \d+ ) ?'
rx = re.compile(numeric_const_pattern, re.VERBOSE)
rx.findall("Some example: Jr. it. was .23 between 2.3 and 42.31 seconds")
如上所述,效果非常好! 不过有一个建议:

re.findall(r"[-+]?\d*\.?\d+|[-+]?\d+", "Current Level: -13.2 db or 14.2 or 3 or -3")

还将返回负的int值(如字符串末尾的-3)

我想您会在我的以下回答中找到有趣的东西,我在前面的类似问题中所做的回答:

在这个答案中,我提出了一个模式,允许正则表达式捕捉任何类型的数字,因为我没有其他可以添加的内容,所以我认为它是相当完整的

scanf() Token      Regular Expression
%e, %E, %f, %g     [-+]?(\d+(\.\d*)?|\.\d+)([eE][-+]?\d+)?
%i                 [-+]?(0[xX][\dA-Fa-f]+|0[0-7]*|\d+)
此正则表达式不支持使用逗号作为整数和小数部分之间的分隔符的国际格式(314159)。 在这种情况下,将上述float regex中的所有
\.
替换为
[,]

                        Regular Expression
International float     [-+]?(\d+([.,]\d*)?|[.,]\d+)([eE][-+]?\d+)?

可以使用以下正则表达式从字符串中获取整数值和浮点值:

re.findall(r'[\d\.\d]+', 'hello -34 42 +34.478m 88 cricket -44.3')

['34', '42', '34.478', '88', '44.3']
谢谢
Rex

它是否总是有整数部分?即使是0?你需要匹配0.4还是0.4?我会说是的。输入是手动输入的,因此有可能出现不一致。“包括4倍大小的AAA 1.5V电池”:-)这些糟糕的用户!总是输入愚蠢的数据。TBH,我故意让这个示例具有演示性,而不是健壮性。当我开始写这个回复时,@MYYN只在接受的答案中提供了正则表达式。我想提供另一种方法的示例。
re.findall(r”[-+]?\d*\.\d+“,“当前级别:-13.2 db或14.2或3”)
['-13.2',14.2',3']
我想您在第一个代码块中的意思是“\d+\.\d+”而不是“\d+.\d+”。现在它将提取类似于“13a4”的内容。@JuanPablo看起来不错,但是
r”[-+]?\d*\。?\d+“
更加简洁,并且不会接受遗漏负整数的
0..4
”—35 um。如果交替在开始时有
[-+]?
“[-+]?\d*\.\d+.[-+]?\d+”
省略了千个分隔符,科学的表达方式,页面上提供了更好的答案非常好!我终于找到了一个非常好的模式!是的,数字的最佳模式。谢谢!在前面添加
(?:\+\s*\-\s*)?
,也会在符号和数字之间留出空间。尽管我承认这可能不是很“标准”,但我在一些文件中看到这种模式“浮动”。在最后一段代码中,您可能需要在模式字符串前面加一个r。这个正则表达式还可以找到句点和数字的非数字组合:
”。。。。1.2.3.4..56..'
产生:
['、'1.2.3.4'、'56..']
re.findall(r'[\d\.\d]+', 'hello -34 42 +34.478m 88 cricket -44.3')

['34', '42', '34.478', '88', '44.3']