Чтение онлайн

ЖАНРЫ

Холзнер Стивен

Шрифт:

LocationPathPattern ::= '/' RelativePathPattern?

 | IdKeyPattern ('/' | '//') RelativePathPattern?

 | '//'? RelativePathPattern

IdKeyPattern ::= 'id' '(' Literal ')' | 'key' '(' Literal '.' Literal ')'

RelativePathPattern ::= StepPattern | RelativePathPattern '/' StepPattern

 | RelativePathPattern '//' StepPattern

StepPattern ::= ChildOrAttributeAxisSpecifier NodeTest Predicate*

ChildOrAttributeAxisSpecifier ::= AbbreviatedAxisSpecifier

 | ('child' | 'attribute') '::'

Определения NodeText (текстового

узла) и Predicate (предиката) приводятся в спецификации XPath (
Expr
соответствует выражению XPath, a
NCName
и
QName
были определены в начале главы 2, «Создание и применение таблиц стилей»):

NodeTest ::= NameTest | NodeType '(' ')' | 'processing-instruction' '(' Literal ')'

Predicate ::= '[' PredicateExpr ']'

PredicateExpr ::= Expr

AbbreviatedAxisSpecifier ::= '@'?

NameTest :: = '*' | NCName ':' '*' | QName

NodeType ::= 'comment' | 'text' | 'processing-instruction' | 'node'

Как вы можете видеть, все это больше походит на какой-то код. Давайте начнем его расшифровывать. Во-первых, образец (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 задаст ему более высокий приоритет для элементов
<MASS>
.

Поделиться с друзьями: