Как извлечь плавающее число из строки

У меня есть ряд строк, похожих на Current Level: 13.4 db. и я хотел бы извлечь только число с плавающей запятой. Я говорю плавающий, а не десятичный, поскольку иногда он цел. Может ли RegEx сделать это или есть лучший способ?

7 ответов

Решение

Если ваш float всегда выражается в десятичной записи что-то вроде

>>> 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

Вы можете попробовать что-то вроде этого, которое охватывает все базы, в том числе не полагаясь на пробел после числа:

>>> 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")

Python docs имеет ответ, который охватывает +/-, и обозначение экспоненты

scanf() Token      Regular Expression
%e, %E, %f, %g     [-+]?(\d+(\.\d*)?|\.\d+)([eE][-+]?\d+)?
%i                 [-+]?(0[xX][\dA-Fa-f]+|0[0-7]*|\d+)

Это регулярное выражение не поддерживает международные форматы, где запятая используется как символ-разделитель между целой и дробной частью (3,14159). В этом случае замените все \. с [.,] в приведенном выше регулярном выражении с плавающей точкой.

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

как описано выше, работает очень хорошо! Одно предложение, хотя:

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

также будет возвращать отрицательные значения типа int (например, -3 в конце этой строки)

Вы можете использовать следующее регулярное выражение для получения целочисленных и плавающих значений из строки:

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

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

Спасибо рекс

Я думаю, что вы найдете интересные вещи в следующем моем ответе, который я сделал для предыдущего аналогичного вопроса:

/questions/47033839/regulyarnoe-vyirazhenie-dlya-sopostavleniya-chisel-s-zapyatyimi-ili-bez-zapyatyih-v-tekste

В этом ответе я предложил шаблон, который позволяет регулярному выражению перехватывать любые числа, и, поскольку мне больше нечего добавить к нему, я думаю, что он достаточно полный.

Другой подход, который может быть более читабельным, - это простое преобразование типов. Я добавил функцию замены, чтобы охватить случаи, когда люди могут вводить европейские десятичные дроби:

>>> 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. Кроме того, предполагается, что люди будут вводить с пробелами между словами. Это не относится к другим языкам, таким как китайский.

Другие вопросы по тегам