</xsl:copy>
</xsl:template>
</xsl:stylesheet>
Однако посмотрите на результат — обратите внимание на то, что в этой версии, выбирающей только элементы и атрибуты (@*|*
), не копируются узлы-разделители и текстовые узлы:
<?xml version='1.0' encoding='UTF-8'?>
<PLANETS><PLANET><NAME/><MASS UNITS='(Earth = 1)'/><DAY UNITS='days'/><RADIUS UNITS='miles'/><DENSITY UNITS='(Earth = 1)'/><DISTANCE UNITS='million miles'/></PLANET><PLANET><NAME/><MASS UNITS='(Earth = 1)'/><DAY UNITS='days'/><RADIUS UNITS='miles'/><DENSITY UNITS='(Earth = 1)'/><DISTANCE UNITS='million miles'/></PLANET><PLANET><NAME/><MASS UNITS='(Earth = 1)'/><DAY UNITS='days'/><RADIUS UNITS='miles'/><DENSITY UNITS='(Earth = 1)'/><DISTANCE UNITS='million miles'/> </PLANET></PLANETS>
Это, конечно, неполно. Если я, с другой стороны, буду выбирать по образцу '@*|node()
' вместо '@*|*
', новое правило шаблона выберет все узлы за исключением корневого узла (который создается в результирующем дереве автоматически), поэтому символы- разделители будут скопированы, так же как и текст (листинг 4.4).
<?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:stylesheet>
Новый результат:
<?xml version='1.0' encoding='UTF-8'?> <?xml-stylesheet type='text/xml' href='planets.xsl'?>
<PLANETS>
<PLANET>
<NAME>Mercury</NAME>
<MASS UNITS='(Earth = 1)'>.0553</MASS>
<DAY UNITS='days'>58.65</DAY>
<RADIUS UNITS='miles'>1516</RADIUS>
<DENSITY UNITS='(Earth = 1)'>.983</DENSITY>
<DISTANCE UNITS='million miles'>43.4</DISTANCE><!--B перигелии-- >
</PLANET>
<PLANET>
<NAME>Venus</NAME>
<MASS UNITS='(Earth = 1)'>.815</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>
.
.
.
Выбор текстовых узлов при помощи text()
Выбрать текст узла можно при помощи образца 'text()
'. Как правило, нет особых причин применять условие узла text
. В XSLT существует правило по умолчанию, в соответствии с которым текст текстового узла включается в выходной документ, если этот узел не выбирается какими-либо другими правилами. Если нужно сделать это правило по умолчанию явным, можно поступить, например, так:
<xsl:template match='text()'>
<xsl:value-of select='.'/>
</xsl:template>
Можно перекрыть это правило, не отправляя текст из текстовых узлов в выходной документ, — в том числе так:
<xsl:template match='text()'>
</xsl:template>
Потребность в применении условия текстового узла возникает, например, когда нужно выбрать узлы с определенным текстом. Предикат 'NAME[text()='Venus']
' выбирает элементы <NAME>
, в которых содержится имя 'Venus
'. (Будьте внимательны с вложением кавычек, чтобы процессор XSLT не ошибся, — например, такой предикат не работает: 'NAME[text()='Venus']
'.) Еще одна причина для использования условия текстового узла появляется, когда требуется применить к текстовым узлам некоторое условие при помощи строковых функций XPath (которые будут рассмотрены позже в этой главе). Например, текстовый узел 'Earth
' в <NAME>Earth</NAME>
выбирается образцом 'text() [starts-with(.,'Е')]
'.
КАК УБРАТЬ КОММЕНТАРИИ
Ранее мы видели, что образец '@*|node()' (в котором используется операция OR, обсуждаемая позже) выбирает из файла planets.xml все, включая комментарии. Если вы хотите убрать комментарии, воспользуйтесь образцом '@*|*|text()', который сохраняет только элементы, атрибуты и текстовые узлы.
Выбор инструкций обработки
Для выбора инструкций обработки используйте образец processing-instruction()
:
<xsl:template match='/processing-instruction()'>
<I>
Found a processing instruction.
</I>
</xsl:template>
Можно также указать, какую именно инструкцию обработки вы хотите выбрать, задав имя инструкции (исключая <?
и ?>
), — как в следующем примере, в котором выбирается