Выражение XPath для выбора всех дочерних узлов XML, за исключением определенного списка?
Здесь примеры данных:
<catalog>
<cd>
<title>Empire Burlesque</title>
<artist>Bob Dylan</artist>
<country>USA</country>
<customField1>Whatever</customField1>
<customField2>Whatever</customField2>
<customField3>Whatever</customField3>
<company>Columbia</company>
<price>10.90</price>
<year>1985</year>
</cd>
<cd>
<title>Hide your heart</title>
<artist>Bonnie Tyler</artist>
<country>UK</country>
<customField1>Whatever</customField1>
<customField2>Whatever</customField2>
<company>CBS Records</company>
<price>9.90</price>
<year>1988</year>
</cd>
<cd>
<title>Greatest Hits</title>
<artist>Dolly Parton</artist>
<country>USA</country>
<customField1>Whatever</customField1>
<company>RCA</company>
<price>9.90</price>
<year>1982</year>
</cd>
</catalog>
Скажем, я хочу выбрать все, кроме элементов цены и года. Я бы хотел написать что-то вроде ниже, что явно не работает.
<?xml version="1.0" encoding="ISO-8859-1"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/">
<html>
<body>
<xsl:for-each select="//cd/* except (//cd/price|//cd/year)">
Current node: <xsl:value-of select="current()"/>
<br />
</xsl:for-each>
</body>
</html>
</xsl:template>
</xsl:stylesheet>
Пожалуйста, помогите мне найти способ исключения определенных дочерних элементов.
Ответы
Ответ 1
<xsl:for-each select="//cd/*[not(self::price or self::year)]">
Но на самом деле это плохо и излишне сложно. Лучше:
<xsl:template match="catalog">
<html>
<body>
<xsl:apply-templates select="cd/*" />
</body>
</html>
</xsl:template>
<!-- this is an empty template to mute any unwanted elements -->
<xsl:template match="cd/price | cd/year" />
<!-- this is to output wanted elements -->
<xsl:template match="cd/*">
<xsl:text>Current node: </xsl:text>
<xsl:value-of select="."/>
<br />
</xsl:template>
Избегайте <xsl:for-each>
. Почти все время это неправильный инструмент и должен быть заменен на <xsl:apply-templates>
и <xsl:template>
.
Вышеупомянутое работает из-за специфичности выражения соответствия. match="cd/price | cd/year"
более специфичен, чем match="cd/*"
, поэтому он является предпочтительным шаблоном для элементов cd/price
или cd/year
. Не пытайтесь исключить узлы, пусть они приходят и обрабатывают их, отбрасывая их.
Ответ 2
Я бы начал экспериментировать с чем-то вроде
"//cd/*[(name() != 'price') and (name() != 'year')]"
Или вы просто выполняете нормальный рекурсивный шаблон с <xsl:apply-templates/>
, а затем имеете пустые шаблоны для элементов <price/>
и <year/>
:
<xsl:template match="price"/>
<xsl:template match="year"/>