Skip to content

Регулярные выражения (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

\cA ... \cZ

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     не пробел

\h

горизонтальный разделитель. Табуляция, пробел и все символы
в Unicode категории "разделители" (space separator Unicode category)
\H не горизонтальный разделитель

\v

вертикальные разделители. новая строка и все символы
"разделители строк" в Unicode
\V не вертикальный разделитель

\R

Юникод разрыв строки: 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 и т. д.

TRegExpr

Свойства 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 нет пустой строки.

TRegExpr

Если вы используете версию 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 перевода строки в неправильном порядке.

TRegExpr

Обработка многострочного текста может быть настроена с помощью свойств 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).

TRegExpr

Для изменения модификаторов используйте 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.

Эти мета-символы также поддерживаются внутри классов символов.

Послесловие

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