JavaScript. Подробное руководство, 6-е издание, стр. 141

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

<b>аЬ</b>
» применяется шаблон
<b>/а|аЬ/</b>
, он будет соответствовать только первому символу.

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

<b>|, *, +, ?</b>
и других рассматриваются как одно целое. Например, шаблон
<b>/java(script)?/</b>
соответствует слову «java», за которым следует необязательное слово «script», a
<b>/(ab|cd)+|ef)/</b>
соответствует либо строке «
<b>ef</b>
», либо одному или более повторений одной из строк «
<b>аЬ</b>
» или «
<b>cd</b>
».

Другим применением скобок в регулярных выражениях является определение подшаблонов внутри шаблона. Когда в целевой строке найдено совпадение с регулярным выражением, можно извлечь часть целевой строки, соответствующую любому конкретному подшаблону, заключенному в скобки. (Мы увидим, как получить эти подстроки, далее в этой главе.) Предположим, что требуется отыскать одну или более букв в нижнем регистре, за которыми следует одна или несколько цифр. Для этого можно воспользоваться шаблоном

<b>/[a-z]+\d+/.</b>
Но предположим также, что нам нужны только цифры в конце каждого соответствия. Если поместить эту часть шаблона в круглые скобки
<b>(/[a-z]+(\d+)/)</b>
, то можно будет извлечь цифры из любых найденных нами соответствий. Как это делается, будет описано ниже.

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

<b>\</b>
. Цифры ссылаются на позицию подвыражения в скобках внутри регулярного выражения. Например,
<b>\1</b>
ссылается на первое подвыражение, а
<b>\3</b>
- на третье. Обратите внимание, что подвыражения могут быть вложены одно в другое, поэтому при подсчете используется позиция левой скобки. Например, в следующем регулярном выражении ссылка на вложенное подвыражение
<b>([Ss]cript)</b>
будет выглядеть как
<b>\2</b>
:

<b>/([Jj]ava([Ss]cript)?)\sis\s(fun\w*)/</b>

Ссылка на предыдущее подвыражение указывает не на шаблон этого подвыражения, а на найденный текст, соответствующий этому шаблону. Поэтому ссылки могут использоваться для наложения ограничения, выбирающего части строки, содержащие точно такие же символы. Например, следующее регулярное выражение соответствует нулю или более символам внутри одинарных или двойных кавычек. Однако оно не требует, чтобы открывающие и закрывающие кавычки соответствовали друг другу (т. е. чтобы обе кавычки были одинарными или двойными):

<b>/['&quot;][~'&quot;]*['&quot;]/</b>

Соответствия кавычек мы можем потребовать посредством такой ссылки:

<b>/(['&quot;])[~'&quot;]*\1/</b>

Здесь

<b>\1</b>
соответствует совпадению с первым подвыражением. В этом примере ссылка налагает ограничение, требующее, чтобы закрывающая кавычка соответствовала открывающей. Это регулярное выражение не допускает присутствия одинарных кавычек внутри двойных, и наоборот. Недопустимо помещать ссылки внутрь классов символов, т. е. мы не можем написать:

<b>/(['&quot;])[~\1]*\1/</b>

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

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

<b>(</b>
и
<b>)</b>
начните группу с символов
<b>(?:</b>
и закончите ее символом
<b>)</b>
. Рассмотрим, например, следующий шаблон:

<b>/([Jj]ava(?:[Ssjcript)?)\sis\s(fun\w*)/</b>

Здесь подвыражение

<b>(?:[Ss]cript)</b>
необходимо только для группировки, чтобы к группе мог быть применен символ повторения ?. Эти модифицированные скобки не создают ссылку, поэтому в данном регулярном выражении \2 ссылается на текст, соответствующий шаблону
<b>(fun\w*).</b>

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

JavaScript. Подробное руководство, 6-е издание - i_016.jpg

10.1.5. Указание позиции соответствия

Как описывалось ранее, многие элементы регулярного выражения соответствуют одному символу в строке. Например,

<b>\s</b>
соответствует одному пробельному символу. Другие элементы регулярных выражений соответствуют позициям между символами, а не самим символам. Например,
<b>\b</b>
соответствует границе слова - границе между
<b>\w</b>
(текстовый ASCII-символ) и
<b>\W</b>
(нетекстовый символ) или границе между текстовым ASCII-символом и началом или концом строки. [20]

Такие элементы, как

<b>\b</b>
, не определяют какие-либо символы, которые должны присутствовать в найденной строке, однако они определяют допустимые позиции для проверки соответствия. Иногда эти элементы называются якорными элементами регулярных выражений, потому что они закрепляют шаблон за определенной позицией в строке. Чаще других используются такие якорные элементы, как
<b>~</b>
и
<b>$</b>
, привязывающие шаблоны соответственно к началу и концу строки.

Например, слово «

<b>JavaScript</b>
», находящееся на отдельной строке, можно найти с помощью регулярного выражения
<b>/~JavaScript$/</b>
. Чтобы найти отдельное слово «
<b>Java</b>
» (а не префикс, например в слове «
<b>JavaScript</b>
»), можно попробовать применить шаблон
<b>/\sJava\s/</b>
, который требует наличия пробела [21] до и после слова. Но такое решение порождает две проблемы. Во-первых, оно найдет слово «Java», только если оно окружено пробелами с обеих сторон, и не сможет найти его в начале или в конце строки. Во-вторых, когда этот шаблон действительно найдет соответствие, возвращаемая им строка будет содержать ведущие и замыкающие пробелы, а это не совсем то, что нам нужно. Поэтому вместо шаблона, совпадающего с пробельными символами
<b>\s</b>
, мы воспользуемся шаблоном (или якорем), совпадающим с границами слова
<b>\b</b>
. Получится следующее выражение:
<b>/\b Java\b/</b>
. Якорный элемент
<b>\В</b>
соответствует позиции, не являющейся границей слова.