Как вы можете видеть, все это больше походит на какой-то код. Давайте начнем его расшифровывать. Во-первых, образец (pattern) состоит из одного (или более) образца пути расположения (location path pattern). Образец пути расположения, в свою очередь, состоит из одного или нескольких образцов шага (step pattern), разделенных / или //, или одним (несколькими) образцом шага в объединении с функциями
id
и
key
(выбирающими элементы с определенными идентификаторами или ключами).
Образцы шага являются строительными блоками шаблонов: в одном пути можно использовать несколько шагов, разделяя их символами / или //, как в образце "
PLANET/*/ NAME
", в котором три шага: "
PLANET
", "
*
" и "
NAME
". Если вы начнете сам образец с символа /, он будет называться абсолютным, так как вы указали образец от корневого узла (как в "
/PLANETS/PLANET
" или "
//PLANET
"); иначе образец называется относительным и применяется начиная с контекстного узла (как в "
PLANET
").
Затем образец шага состоит из оси, условия узла и предикатов (которых может и не быть). Например, в выражении
child::PLANET[position=5]
,
child
— это имя оси,
PLANET
— условие узла, a
[position=5]
— это предикат. (Предикаты всегда заключены в квадратные скобки.) Образцы можно создавать при помощи одного или более образцов шага, как, например, образец
/child::PLANET/child::NAME
, который выбирает элементы
<NAME>
, дочерние по отношению к родителю
<PLANET>
.
Таким образом, чтобы понять работу образцов, вам необходимо понять работу образцов шага, поскольку образцы состоят из одного или более образцов шага, в таких выражениях, как "
step-pattern1/step-pattern2/step-pattern3
…". А чтобы понять работу образца шага, необходимо понять работу деятельности трех составных частей — осей, условий узлов и предикатов, которыми мы и займемся в следующих разделах.
Образцы шага, часть 1: оси образца
Оси — первая часть образцов шага. Например, в образце шага
child::NAME
, ссылающемся на элемент
<NAME>
, дочерний по отношению к контекстному узлу,
child
называется осью. У образцов две оси:
• ось
attribute
содержит атрибуты контекстного узла;
• ось
child
содержит детей контекстного узла. Если ось явно не задана, ось
child
будет осью по умолчанию.
При помощи осей можно задать шаг расположения (location path) или путь, как в следующем примере, в котором ось
child
используется для задания выбора дочерних узлов контекстного узла, элемента
<PLANET>
:
<xsl:template match="PLANET">
<HTML>
<CENTER>
<xsl:value-of select="child::NAME"/>
</CENTER>
<CENTER>
<xsl:value-of select="child::MASS"/>
</CENTER>
<CENTER>
<xsl:value-of select="child::DAY"/>
</CENTER>
</HTML>
</xsl:template>
Рассмотрим ряд примеров применения осей:
•
child::PLANET
. Возвращает дочерние элементы
<PLANET>
контекстного узла;
•
child::*
. Возвращает все дочерние элементы контекстного узла (* выбирает только элементы);
•
attribute::UNITS
. Возвращает атрибут
UNITS
контекстного узла;
•
child::*/child::PLANET
. Возвращает всех внуков
<PLANET>
контекстного узла.
Хотя, судя по этим примерам, кажется, что можно применять только оси детей и атрибутов, на практике это не совсем так. Когда требуется указать детей, возможности оси
child
несколько ограничены, потому что необходимо указывать каждый уровень, который необходимо выбрать — например "
child::PLANETS/child::PLANET/child::MASS
" выбирает элемент
<MASS>
, дочерний по отношению к элементу
<PLANET>
, который, в свою очередь, дочерний по отношению к
<PLANETS>
. Если вам требуется выбрать все элементы
<MASS>
, появляющиеся в любом месте элемента
<PLANETS>
, детей, внуков, правнуков и т.д., кажется, что нет способа сделать это в одном образце. В XPath это можно сделать при помощи выражения наподобие "
child::PLANETS/descendant::MASS
", но в образцах нельзя использовать ось потомков (descendant). Помните, однако, что в этих же целях можно применить операцию
//
. Например, образец "
child::PLANETS//child::MASS
" выбирает все элементы
<MASS>
в любом месте внутри элемента
<PLANETS>
.
Следующий пример (листинг 4.2) демонстрирует работу этого образца, заменяя текст во всех элементах
<MASS>
независимо от того, где они находятся внутри элемента
<PLANETS>
, на текст "
Very heavy!
". Для того чтобы скопировать в результирующий XML-документ все остальные узлы
planets.xml
, я также установил правило, выбирающее любой узел при помощи условия узла (node test)
node
, с которым мы познакомимся позже. Заметьте, что, хотя образец, выбирающий любой узел, также выбирает все элементы
<MASS>
, образец "
child::PLANETS//child::MASS
" гораздо более специален — поэтому, как объяснялось в главе 3, процессор XSLT задаст ему более высокий приоритет для элементов