элементы);
• 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>.
<?xml version='1.0'?>
<xsl:stylesheet version='1.0'
xmlns:xsl='http://www.w3.org/1999/XSL/Transform'>
<xsl:output method='xml'/>
<xsl:template match='@*|node()'>
<xsl:copy>
<xsl:apply-templates select='@*|node()'/>
</xsl:copy>
</xsl:template>
<xsl:template match='child::PLANETS//child::MASS'>
<MASS>
Very heavy!
</MASS>
</xsl:template>
</xsl:stylesheet>
А вот результирующий XML-документ:
<?xml version='1.0' encoding-'UTF-8'?>
<?xml-stylesheet type='text/xml' href='planets.xsl'?>
<PLANETS>
<PLANET>
<NAME>Mercury</NAME>
<MASS>
Very heavy!
</MASS>
<DAY UNITS='days'>58.65</DAY>
<RADIUS UNITS='miles'>1516</RADIUS>
<DISTANCE UNITS='million miles'>43.4</DISTANCE><!--В перигелии- >
</PLANET>
<PLANET>
<NAME>Venus</NAME>
<MASS>
Very heavy!
</MASS>
<DAY UNITS='days'>116.75</DAY>
<RADIUS UNITS='miles'>3716</RADIUS>
<DENSITY UNITS='(Earth = 1)'>.943</DENSITY>
<DISTANCE UNITS='million miles'>66.8</DISTANCE><!--B перигелии-- >
</PLANET>
<PLANET>
<NAME>Earth</NAME>
<MASS>
Very heavy!
</MASS>
<DAY UNITS='days'>1</DAY>
<RADIUS UNITS='miles'>2107</RADIUS>
<DENSITY UNITS='(Earth = 1)'>1</DENSITY>
<DISTANCE UNITS='million miles'>128.4</DISTANCE><!--B перигелии-- >
</PLANET>
</PLANETS>
При задании осей в образцах можно воспользоваться рядом сокращений, применяемых практически повсеместно.
Сокращенный синтаксис
Для образцов существует два правила сокращения осей:
• child::childname может быть сокращено как childname;
• attribute::childname может быть сокращено как @childname.
В следующем списке перечислен ряд примеров образцов с сокращенным синтаксисом; в конце главы вы увидите много других.
• PLANET. Выбирает дочерние элементы <PLANET> контекстного узла;
• *. Выбирает все дочерние элементы контекстного узла;
• @UNITS. Выбирает атрибут UNITS узла;
• @*. Выбирает все атрибуты контекстного узла;
• */PLANET. Выбирает всех внуков <PLANET> контекстного узла;
