我有一个XML文件,例如:
<doc>
<element>
<word>
<text>one word</text>
</word>
<word>
<text>two words</text>
</word>
</element>
<element>
<other>
<text>the sky</text>
</other>
<word>
<text>NN NN</text>
</word>
</element>
</doc>
,我只想有一个标签,当两个之后有两个标签时,有两个标签,例如:
<element>
<word>
<text>one word</text>
<text>two words</text>
</word>
</element>
问题是我正在生成<xsl:element>
以获取<word>
标签。并且此<xsl:element>
已经在<xsl:for-each>
循环中,该循环将<word>
标签以正确的顺序(在属性上过滤)。
我以前的XML文档就是这样:
<doc>
<element class="word">
<text>one word</text>
</element>
<element class="word">
<text>NN NN</text>
</element>
<element class="word">
<text>two words</text>
</element>
<element class="other">
<text>the sky</text>
</element>
</doc>
任何帮助都会很棒,谢谢:)
- 更多详细信息 -
这是我想拥有的结果:
<doc>
<element>
<word>
<text>one word</text>
<text>two words</text>
</word>
</element>
<element>
<other>
<text>the sky</text>
</other>
<word>
<text>NN NN</text>
</word>
</element>
</doc>
我无法指定标签 Word 或其他,因为我没有标签的封闭列表。因此,我正在使用 XSL:变量:
<xsl:for-each select="element">
<xsl:variable name="name">
<xsl:value-of select="@class"/>
</xsl:variable>
<xsl:element name="{$name}">
<text>
<xsl:value-of select="."/>
</text>
</xsl:element>
</xsl:for-each>
尝试以下:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="element/*">
<xsl:if test="not(preceding-sibling::*[name() = name(current())])">
<xsl:copy>
<xsl:apply-templates select="* | following-sibling::*[name()=name(current())]/*" />
</xsl:copy>
</xsl:if>
</xsl:template>
<!-- Identity template -->
<xsl:template match="@* | node()">
<xsl:copy>
<xsl:apply-templates select="@* | node()"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
它只是在查找没有以前具有相同名称的兄弟姐妹(即每个孩子的第一个)的element
的任何子节点以同名。
您只需使用<xsl:apply-templates/>
...
XML输入
<doc>
<element>
<word>
<text>one word</text>
</word>
<word>
<text>two words</text>
</word>
</element>
<element>
<other>
<text>the sky</text>
</other>
<word>
<text>NN NN</text>
</word>
</element>
</doc>
XSLT 1.0
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="element">
<xsl:copy>
<word>
<xsl:apply-templates/>
</word>
</xsl:copy>
</xsl:template>
<xsl:template match="word|other">
<xsl:apply-templates/>
</xsl:template>
</xsl:stylesheet>
输出
<doc>
<element>
<word>
<text>one word</text>
<text>two words</text>
</word>
</element>
<element>
<word>
<text>the sky</text>
<text>NN NN</text>
</word>
</element>
</doc>
此转换:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="node()|@*">
<xsl:copy>
<xsl:apply-templates select="node()|@*"/>
</xsl:copy>
</xsl:template>
<xsl:template match="element/*"/>
<xsl:template match="element/*[1]">
<word>
<xsl:apply-templates select="../*/*"/>
</word>
</xsl:template>
</xsl:stylesheet>
应用于提供的XML文档:
<doc>
<element>
<word>
<text>one word</text>
</word>
<word>
<text>two words</text>
</word>
</element>
<element>
<other>
<text>the sky</text>
</other>
<word>
<text>NN NN</text>
</word>
</element>
</doc>
生产(我猜是)想要的,正确的结果:
<doc>
<element>
<word>
<text>one word</text>
<text>two words</text>
</word>
</element>
<element>
<word>
<text>the sky</text>
<text>NN NN</text>
</word>
</element>
</doc>