Регулярные выражения (RegEx)
Вступление
Регулярные выражения являются простым и мощным инструментом для сложного поиска и замены, а также для проверки текста на основе шаблонов.
Ниже приведена исчерпывающая шпаргалка по регулярных выражениям всего на одной странице.
Символы
Простые совпадения
Любой отдельный символ (кроме специальных символов регулярных выражений) совпадает сам с собой. Последовательность символов (не специальных) совпадает с такой же последовательностью символов во входной строке.
RegEx | Находит |
---|---|
foobar |
foobar |
Непечатаемые символы (escape-коды)
Чтобы указать символ по его Unicode коду, используйте префикс \x
, за которым следует
шестнадцатеричный код.
Для кода из 3-4 цифр (после U+00FF) заключите код в фигурные скобки.
RegEx | Находит |
---|---|
\xAB |
символ с 2-значным шестнадцатеричным кодом AB |
\x{AB20} |
символ с 1-4 значным шестнадцатеричным кодом AB20 |
foo\x20bar |
foo bar (обратите внимание на пробел в середине) |
Существует ряд предопределённых escape-кодов для непечатаемых символов, как в языке C:
RegEx | Находит |
---|---|
\t |
tab (HT/TAB), тоже что \x09 |
\n |
символ новой строки (LF), то же что \x0a |
\r |
возврат каретки (CR), тоже что \x0d |
\f |
form feed (FF), то же что \x0c |
\a |
звонок (BEL), тоже что \x07 |
\e |
escape (ESC), то же что \x1b |
|
chr(0) по chr(25). Например \cI соответствует табуляции.Также поддерживаются буквы в нижнем регистре "a"..."z". |
Экранирование(escaping)
Чтобы представить специальный символ регулярного выражения (один из .+*?|\()[]{}^$
),
поставьте перед ним обратный слэш \
.
Сам обратный слэш также должен быть экранирован.
RegEx | Находит |
---|---|
\^FooBarPtr |
^FooBarPtr здесь ^ не означает начало строки |
\[a\] |
[a] это не класс символов |
Классы символов
User Классы символов
Класс символов - это список символов внутри квадратных скобок []
.
Класс совпадает с любым одиночным символом, указанным в этом классе.
RegEx | Находит |
---|---|
foob[aeiou]r |
foobar , foober и т. д., но не foobbr , foobcr и т. д. |
Вы можете "инвертировать" класс - если первым символом после [
является ^
,
тогда класс совпадает с любым символом, кроме символов, перечисленных в классе.
RegEx | Находит |
---|---|
foob[^aeiou]r |
foobbr , foobcr и т. д., но не foobar , foober и т. д. |
Внутри списка символ тире -
используется для указания диапазона, так что a-z
представляет все символы между a
и z
, включая их.
Если вы хотите, чтобы само тире -
было элементом класса, поместите его в начало или в
конец списка, или экранируйте его обратным слэшем.
Если вам нужен символ ]
в качестве части класса, вы можете разместить его в начале
списка или экранируйте его обратным слэшем.
RegEx | Находит |
---|---|
[-az] |
a , z и - |
[az-] |
a , z и - |
[А\-z] |
a , z и - |
[a-z] |
символы от a до z |
[\n-\x0D] |
символы от #10 до #13 |
Метасимвол .
(точка)
Метасимвол .
(точка) по умолчанию совпадает с любым символом.
Но если вы выключите модификатор /s, тогда он не будет совпадать с символами
переноса строки.
Символ .
не действует как мета-класс внутри
пользовательских классов символов.
[.]
означает буквальную точку.
Метаклассы
Существует ряд предопределённых классов символов, которые делают регулярные выражения более компактными, "мета-классы":
RegEx | Находит |
---|---|
\w |
буквенно-цифровой символ (включая _ ) |
\W |
не буквенно-цифровой |
\d |
числовой символ (тоже, что [0-9] ) |
\D |
нечисловой |
\s |
любой пробел (такой же как [\t\n\r\f] ) |
\S |
не пробел |
|
горизонтальный разделитель. Табуляция,
пробел и все символы в Unicode категории "разделители" (space separator Unicode category) |
\H |
не горизонтальный разделитель |
|
вертикальные разделители. новая строка и все
символы "разделители строк" в Unicode |
\V |
не вертикальный разделитель |
|
Юникод разрыв строки: LF, пара CR LF, CR. FF (form feed), VT (vertical tab), U+0085, U+2028, U+2029 |
Вы можете использовать все мета-классы, упомянутые в таблице выше, внутри пользовательских классов символов.
RegEx | Находит |
---|---|
foob\dr |
foob1r , foob6r и т. д., но не foobar , foobbr и т. д. |
foob[\w\s]r |
foobar , foob r , foobbr и т. д., но не foob1r , foob=r и т. д. |
Свойства SpaceChars и WordChars определяют классы символов
\w
,\W
,\s
,\S
.Таким образом, вы можете переопределить эти классы.
Разделители
Разделители строк
Метасимвол | Находит |
---|---|
^ |
совпадение нулевой длины в начале строки |
$ |
совпадение нулевой длины в конце строки |
\A |
совпадение нулевой длины в начале строки |
\z |
совпадение нулевой длины в конце строки |
\Z |
похож на \z но совпадает перед разделителем строки, а не сразу после него, как \z |
\G |
совпадение нулевой длины в конечной позиции предыдущего совпадения |
Примеры:
RegEx | Находит |
---|---|
^foobar |
foobar только если он находится в начале строки |
foobar$ |
foobar , только если он в конце строки |
^foobar$ |
foobar только если это единственная строка в строке |
foob.r |
foobar , foobbr , foob1r и так далее |
Метасимвол ^
совпадает с позицией нулевой длины в начале входной строки. $
- в конце.
Если модификатор /m включен, они также совпадают с началом/концом отдельных
строк в многострочном тексте.
Обратите внимание, что в последовательности \x0D\x0A
нет пустой
строки.
Если вы используете версию Unicode, то
^
/$
также совпадают с\x2028
,\x2029
,\x0B
,\x0C
или\x85
.
Метасимвол \A
совпадает с позицией нулевой длины в самом начале входной строки,
\z
- в самом конце.
Они игнорируют модификатор /m. \Z
похож на \z
, но совпадает перед
разделителем строки, а не сразу после него, как \z
(LF и CR LF).
Поведение \A
, \z
, \Z
реализовано подобно большинству движков regex
(Perl, PCRE и т.д.).
Отметим, что ^.*$
не совпадает со строкой между \x0D\x0A
, потому что это
неразрывный разделитель строк.
Но оно совпадает с пустой строкой в последовательности \x0A\x0D
, потому что это
2 перевода строки в неправильном порядке.
Обработка многострочного текста может быть настроена с помощью свойств LineSeparators и UseLinePairedBreak.
Таким образом, вы можете использовать разделители в стиле Unix
\n
или DOS/Windows\r\n
или смешивать их вместе (как в описанном выше поведении по умолчанию).
Если вы предпочитаете математически точное описание, вы можете найти его на www.unicode.org.
Разделители слов
RegEx | Находит |
---|---|
\b |
разделитель слов |
\B |
разделитель с не-словом |
Граница слова \b
- это место между двумя символами, где с одной стороны находится
\w
, а с другой - \W
(в любом порядке).
Повторы
Повтор
Любой элемент регулярного выражения может быть снабжён квантификатором. Квантификатор указывает количество повторений элемента.
RegEx | Находит |
---|---|
{n} |
ровно n раз |
{n,} |
по крайней мере n раз |
{,m} |
not more than m times (only with AllowBraceWithoutMin) |
{n,m} |
по крайней мере n , но не более чем m раз |
* |
ноль или более, аналогично {0,} |
+ |
один или несколько, похожие на {1,} |
? |
ноль или единица, похожая на {0,1} |
Так, цифры в фигурных скобках {n,m}
указывают минимальное количество совпадений
n
и максимальное m
.
{n}
эквивалентен {n,n}
и совпадает ровно n
раз. {n,}
совпадает n
или более раз.
Вариант {,m}
поддерживается только если установлено свойство AllowBraceWithoutMin.
Теоретически значение n и m не ограничены (можно использовать максимальное значение для 32-х битного числа).
Использование {
без указания корректного диапазона приведет к ошибке.
Это поведение может быть изменено установкой свойства AllowLiteralBraceWithoutRange,
которое позволит принять {
как буквальный символ, если за ним не следует диапазон.
Диапазон, в котором нижнее значение больше верхнего, всегда приведет к ошибке.
RegEx | Находит |
---|---|
foob.*r |
foobar , foobalkjdflkj9r и foobr |
foob.+r |
foobar , foobalkjdflkj9r , но не foobr |
foob.?r |
foobar , foobbr и foobr , но не foobalkj9r |
fooba{2}r |
foobaar |
fooba{2}r ` |
foobaar, foobaaar, foobaaaar` и т. д. |
fooba{2,3}r |
foobaar , или foobaaar , но не foobaaaar |
(foobar){8,10} |
8 , 9 или 10 экземпляров foobar (() это Группа) |
Жадность
Повтор в "жадном" режиме берет столько, сколько возможно, в "ленивом" режиме - как можно меньше.
По умолчанию все квантификаторы "жадные".
Добавьте символ ?
, чтобы сделать любой квантификатор "ленивым".
Для строки abbbbc
:
RegEx | Находит |
---|---|
b+ |
bbbb |
Ь+? |
b |
b*? |
пустую строку |
b{2,3}? |
bb |
b{2,3} |
bbb |
Вы можете переключить все квантификаторы в "ленивый" режим (модификатор /g, ниже мы используем изменение модификатора в строке).
RegEx | Находит |
---|---|
(?-g)Ь+ |
b |
Сверхжадные повторы (Possessive Quantifier)
Синтаксис следующий: a++
, a*+
, a?+
, a{2,4}+
.
В настоящее время это поддерживается только для простых скобок, но не для скобок
после группы, как в (foo|bar){3,5}+
.
Эта функция регулярных выражений описана здесь. Коротко говоря, владеющий квантификатор ускоряет совпадение в сложных случаях.
Альтернативы
Выражения в списке альтернатив разделяются |
.
Таким образом, fee|fie|foe
совпадет с любым из fee
, fie
или foe
в
целевой строке (как и f(e|i|o)e
).
Первое выражение включает все, начиная от последнего разделителя шаблона ((
, [
или начала шаблона) до первого |
, а последнее выражение содержит все от последнего |
до следующего разделителя шаблона.
Звучит немного сложно, поэтому обычно выбор заключают в скобки, чтобы минимизировать путаницу относительно его начала и конца.
Выражения в выборе пробуются слева направо, поэтому выбирается первое совпадающее выражение.
Например, регулярное выражение foo|foot
в строке barefoot
совпадет с foo
.
Просто с первым совпадающим выражением.
Также помните, что |
интерпретируется как буквальный символ в квадратных скобках,
так что если вы напишете [fee|fie|foe]
, на самом деле вы совпадете только с [feio|]
.
RegEx | Находит |
---|---|
foo(bar|foo) |
foobar или foofoo |
Группы (подвыражения)
Скобки (...)
также могут использоваться для определения групп
(подвыражений) регулярного выражения.
[!NOTE] TRegExpr
Группы (подвыражения) позиции, длины и подвыражения будут в MatchPos, MatchLen и Match.
Их можно заменить с помощью Substitute.
Группы (подвыражения) нумеруются слева направо по их открывающей скобке (включая вложенные группы). Первая группа имеет индекс 1. Весь регулярное выражение имеет индекс 0.
Группы (подвыражения) значение 0 foobar
1 foobar
2 bar
Ссылки на группы (Backreferences)
Метасимволы \1
до \9
интерпретируются как обратные ссылки (backreference) на
группы захвата. Они соответствуют ранее найденной группе с указанным
индексом.
Метасимвол \g
, за которым следует число, также интерпретируется как
обратная ссылка на группы захвата. За ним может следовать число из нескольких цифр.
RegEx | Находит |
---|---|
(.)\1+ |
aaaa и cc |
(.+)\1+ |
также abab и 123123 |
(.)\g1+ |
aaaa и cc |
RegEx (['"]?)(\d+)\1
соответствует "13"
(в двойных кавычках), или '4'
(в
одинарных кавычках) или 77
(без кавычек) и т. д.
Именованные группы (подвыражения) and Ссылки на группы (Backreferences)
Чтобы присвоить имя группе expr
используйте (?P<name>expr)
или (?<name>expr)
или
(?'name'expr)
.
Имя группы должно начинаться с буквы или _
, далее следуют буквы, цифры
или _
. Именованные и не именованные группы имеют общую нумерацию от
1
до 9
.
Ссылки на именованные группы (Backreferences) обозначаются как (?P=name)
,
также можно использовать числа от \1
до \9
. Также можно использовать (?P=name)
.
Пример
RegEx | Находит |
---|---|
(?P<qq>['"])\w+(?P=qq) |
"word" и 'word' |
Результат соответствия (match)
Начало сообщаемого соответствия можно установить с помощью \K
.
По умолчанию весь текст, покрываемый шаблоном, считается соответствующим. Однако можно явно указать, что будет сообщено.
Шаблон a\Kb
требует, чтобы текст содержал "ab". Но только "b" будет
сообщено как соответствующее. В шаблоне может быть несколько \K
.
Последний установит позицию начала соответствия. Рассматриваются только \K
в активных частях шаблона. Например, a(\Kb)?
не будет учитывать \K
, если
нет "b". Захваты могут существовать вне соответствия, установленного через \K
.
Если используется в других конструкциях, которые могут применяться вне сообщаемого соответствия
(например, в просмотре вперед), то позиция, отмеченная \K
, должна быть до или в
конце сообщаемого соответствия. Если позиция отмечена позже, соответствие считается неудачным.
\K
несколько похож на просмотр назад. В отличие от просмотра назад, часть
шаблона перед \K
должна находиться после начальной позиции соответствия, если шаблон применяется
со смещенной позиции внутри текста.
Модификаторы
Модификаторы используются для изменения поведения регулярных выражений.
Вы можете установить модификаторы глобально в вашей системе или изменить внутри регулярного выражения, используя (?imsxr-imsxr).
Для изменения модификаторов используйте ModifierStr или соответствующие свойства
TRegExpr
Modifier*.Значения по умолчанию определены в глобальных переменных. Например, глобальная переменная
RegExprModifierX
определяет значение по умолчанию для свойстваModifierX
.
i, без учета регистра
Без учета регистра. Использует настройки локали, установленные в вашей системе, см. также InvertCase.
m, многострочные строки
Рассматривать строку как несколько строк. Таким образом, ^
и $
соответствуют началу или концу
любой строки где угодно внутри строки.
См. также Разделители строк.
s, одиночные строки
Рассматривать строку как одну строку. Таким образом, .
соответствует любому символу,
включая разделители строк.
См. также Разделители строк, которым обычно не соответствовал бы.
г, жадность
Специфичный для TRegExpr модификатор.
Выключив его, вы переключите квантификаторы в нежадный режим.
Так что, если модификатор /g
выключен, то +
работает как +?
, *
как *?
и так
далее.
По умолчанию этот модификатор имеет значение Выкл
.
x, расширенный синтаксис
Позволяет комментировать регулярное выражение и разбивать его на несколько строк.
Если модификатор включен, мы игнорируем все пробелы, которые не являются экранированными или находятся внутри класса символов.
Также символ #
отделяет комментарии.
Обратите внимание, что вы можете использовать пустые строки для форматирования регулярного выражения для лучшей читаемости:
(
(abc) # comment 1
#
(efg) # comment 2
)
Это также означает, что если вам нужны настоящие пробелы или символы #
в
шаблоне (вне класса символов, где они не подвержены влиянию /x
), вам придется либо
экранировать(escape) их, либо кодировать с помощью октальных или шестнадцатеричных
escape-последовательностей.
г, русские диапазоны
Специфичный для TRegExpr модификатор.
В русской таблице ASCII символы ё
/Ё
расположены отдельно от
остальных.
Большие и маленькие русские буквы находятся в разных диапазонах, это так же, как и с английскими буквами, но тем не менее мне хотелось бы краткой формы.
С этим модификатором вместо [а-яА-ЯёЁ]
вы можете написать [а-Я]
, если вам
нужны все русские символы.
Когда модификатор включен:
RegEx | Находит |
---|---|
а-я |
символы от а до я и ё |
А-Я |
символы от А до Я и Ё |
а-Я |
все русские символы |
Модификатор по умолчанию установлен на Вкл
.
Проверки или заглядывания вперед и назад (assertions: lookahead, lookahead)
Заглядывание вперед (lookahead assertion) foo(?=bar)
совпадает "foo"
только перед "bar", при этом сама строка "bar" не войдет в найденный
текст.
Отрицательное заглядывание вперед (negative lookahead assertion):
foo(?!bar)
совпадает "foo" только если после этой строки не следует
"bar".
Ретроспективная проверка (lookbehind assertion): (?<=foo)bar
совпадает
"bar" только после "foo", при этом сама строка "foo" не войдет в
найденный текст.
Отрицательное заглядывание назад (negative lookbehind assertion):
foo(?!bar)
совпадает "bar" только если перед этой строкой нет "foo".
Ограничения:
- Переменная длина lookbehind не допускает содержание групп захвата.
Это можно разрешить, установив свойство
AllowUnsafeLookBehind
. Если это включено и в тексте есть более одного соответствия, которое группа может захватить, то может быть захвачено неправильное соответствие. Это не влияет на правильность общего утверждения. (Т.е., lookbehind корректно вернет, если текст перед ним соответствует шаблону). - Переменная длина lookbehind может быть медленной в исполнении, если они не соответствуют.
Не захватывающие группы (подвыражения)
Синтаксис: (?:subexpression)
.
Такие группы не имеют "индекса" и невидимы для обратных ссылок. Незахватывающие группы используются, когда вы хотите сгруппировать подвыражение, но не хотите сохранять его как совпавшую/захваченную часть строки. Таким образом, это просто способ организовать ваше регулярное выражение в подвыражения без накладных расходов на захват результата:
RegEx | Находит |
---|---|
(https?|ftp)://([^/\r\n]+) |
in https://sorokin.engineer matches https and sorokin.engineer |
(?:https?|ftp)://([^/\r\n]+) |
in https://sorokin.engineer matches only sorokin.engineer |
Атомарные группы (подвыражения)
Синтаксис: (?>expr|expr|...)
.
Атомарные группы — это особый случай незахватывающих групп. Описание их.
Inline Модификаторы
Синтаксис для одного модификатора: (?i)
чтобы включить, и (?-i)
чтобы выключить. Для большого числа модификаторов используется
синтаксис: (?msgxr-imsgxr)
.
Вы можете использовать это внутри регулярного выражения для изменения модификаторов
на лету.
Это может быть особенно удобно, потому что оно имеет локальную область видимости в
регулярном выражении.
Оно влияет только на ту часть регулярного выражения, которая следует за оператором
(?imsgxr-imsgxr)
.
И если это внутри группы, это будет влиять только на эту группу - конкретно на часть
группы, которая следует за модификаторами.
Так в ((?i)Saint)-Petersburg
это влияет только на группу ((?i)Saint)
,
поэтому оно будет соответствовать saint-Petersburg
, но не saint-petersburg
.
Встроенные модификаторы также могут быть заданы как часть незахватывающей группы:
(?i:pattern)
.
RegEx | Находит |
---|---|
(?i)Saint-Petersburg |
Saint-petersburg и Saint-Petersburg |
(?i)Saint-(?-i)Petersburg |
Saint-Petersburg , но не Saint-petersburg |
(?i)(Saint-)?Petersburg |
Saint-petersburg и saint-petersburg |
((?i)Saint-)?Petersburg |
saint-Petersburg , но не saint-petersburg |
Комментарии
Синтаксис: (?#text)
. Все, что внутри скобок, игнорируется.
Обратите внимание, что комментарий закрывается ближайшей )
,
поэтому нет способа поместить литеральную )
в комментарий.
Рекурсия
Синтаксис (?R)
, синоним (?0)
.
Выражение a(?R)?z
совпадает с одним или более символом "a" за которым
следует точно такое же число символов "z".
Основное назначение рекурсии - сбалансировать обрамление вложенного
текста. Общий вид b(?:m|(?R))*e
где "b" это то что начинает
обрамляемый текст, "m" это собственно текст, и "e" это то, что завершает
обрамление.
Если же обрамляемый текст также может встречаться без обрамления то
выражение будет b(?R)*e|m
.
Вызовы подвыражений
Нумерованные группы (подвыражения) обозначают (?1)
... (?90)
(максимальное число групп определяется константой в TRegExpr).
Синтаксис для именованных групп : (?P>name)
. Поддерживается также: (?&name)
, \g<name>
и \g'name'
.
Это похоже на рекурсию, но повторяет только указанную группу (подвыражение).
Unicode категории (category)
В стандарте Unicode есть именованные категории символов (Unicode category). Категория обозначается одной буквой, и еще одна добавляется, чтобы указать подкатегорию. Например "L" это буква в любом регистре, "Lu" - буквы в верхнем регистре, "Ll" - в нижнем.
- Cc - Control
- Cf - Формат
- Co - Частное использование
- Cs - Заменитель (Surrrogate)
- Ll - Буква нижнего регистра
- Lm - Буква-модификатор
- Lo - Прочие буквы
- Lt - Буква в начальном регистре
- Lu - Буква в верхнем регистре
- Mc - Разделитель
- Me - Закрывающий знак (Enclosing Mark)
- Mn - Несамостоятельный символ, как умляут над буквой (Nonspacing Mark)
- Nd - Десятичная цифра
- Nl - Буквенная цифра - например, китайская, римская, руническая и т.д. (Letter Number)
- No - Другие цифры
- Pc - Connector Punctuation
- Pd - Тире
- Pe - Закрывающая пунктуация
- Pf - Конечная пунктуация
- Pi - Начальная пунктуация
- Po - Прочая пунктуация
- Ps - Открывающая пунктуация
- Sc - Символ валюты
- Sk - Символ-модификатор
- Sm - Математический символ
- So - Прочие символы
- Zl - Разделитель строк
- Zp - Разделитель абзацев
- Zs - Разделитель пробелов
Метасимволacter \p
denotes one Unicode char of specified category.
Syntax: \pL
and \p{L}
for 1-letter name, \p{Lu}
for 2-letter
names.
Метасимволacter \P
is inverted, it denotes one Unicode char not in
the specified category.
Эти мета-символы также поддерживаются внутри классов символов.
Послесловие
В этом древнем блог-посте из прошлого века я иллюстрирую некоторые способы использования регулярных выражений.