許多很多解決方案。這一次使用細粒度遍歷:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="node()|@*" name="identity">
<xsl:copy>
<xsl:apply-templates select="node()[1]|@*"/>
</xsl:copy>
<xsl:apply-templates select="following-sibling::node()[1]"/>
</xsl:template>
<xsl:template match="bar">
<foo>
<xsl:call-template name="identity"/>
</foo>
<xsl:apply-templates select="following-sibling::bar[1]"/>
</xsl:template>
<xsl:template match="qux">
<xsl:copy>
<xsl:apply-templates select="node()[1]|@*"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
輸出:
<root>
<foo>
<bar>bar 1</bar>
<baz>baz 1</baz>
<qux>qux 1</qux>
</foo>
<foo>
<bar>bar 2</bar>
<baz>baz 2</baz>
<qux>qux 2</qux>
</foo>
</root>
其他的解決方案:用的按鍵風格。
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:key name="kElementByPrecedingBar" match="root/*[not(self::bar)]"
use="generate-id(preceding-sibling::bar[1])"/>
<xsl:template match="node()|@*" name="identity">
<xsl:copy>
<xsl:apply-templates select="node()|@*"/>
</xsl:copy>
</xsl:template>
<xsl:template match="bar" mode="wrap">
<foo>
<xsl:apply-templates select=".|key('kElementByPrecedingBar',
generate-id())"/>
</foo>
</xsl:template>
<xsl:template match="root">
<xsl:copy>
<xsl:apply-templates select="bar" mode="wrap"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
好問題(+1)。看到我的答案是一個有效的解決方案,同時它是通用的,並且沒有任何元素名稱硬編碼。 :) – 2010-09-30 15:31:01