给定以下XML:
<root>
<group>
<e1>001</e1>
<e2>beep</e2>
<e2>bop</e2>
<e2>ork</e2>
<e2>ah</e2>
<e2>ah</e2>
</group>
<group>
<e1>002</e1>
<e2>beep</e2>
<e2>bop</e2>
<e2>ork</e2>
<e2>ah</e2>
<e2>ah</e2>
</group>
<group>
<e1>003</e1>
<e2>beep</e2>
<e2>bop</e2>
<e2>ork</e2>
<e2>ah</e2>
<e2>ah</e2>
</group>
<group>
<e1>004</e1>
<e2>beep</e2>
<e2>bop</e2>
<e2>ork</e2>
<e2>ah</e2>
<e2>ah</e2>
</group>
</root>
请注意,每个“组”元素中的“e2”元素都是相同的,这在源文档中是有保证的。
我正在尝试使用XSLT来完成以下步骤:
所需的输出将如下所示:
<root>
<group>
<e1>default1</e1>
<e2>beep</e2>
<e2>bop</e2>
<e2>ork</e2>
<e2>ah</e2>
<e2>ah</e2>
</group>
<group>
<e1>default2</e1>
<e2>beep</e2>
<e2>bop</e2>
<e2>ork</e2>
<e2>ah</e2>
<e2>ah</e2>
</group>
</root>
源文档中的'e1‘值是不相关的,而输出文档中的'e2’值是预先知道的,并且是静态的。只有'e2‘值是动态的,我需要确保它们都在那里。
我以前在用一些硬编码值替换所有元素时使用过类似的模式:
<xsl:template match="node() | @*">
<xsl:copy>
<xsl:apply-templates select="node() | @*"/>
</xsl:copy>
</xsl:template>
<!-- Empty Template eliminates all but first 'group' element. -->
<xsl:template match="//group[preceding::group]"></xsl:template>
<xsl:template match="//group">
<xsl:element name="group">
<e1>default1</e1>
<!-- e2 elements inserted here somehow? -->
</xsl:element>
<xsl:element name="group">
<e1>default2</e1>
<!-- e2 elements inserted here somehow? -->
</xsl:element>
</xsl:template>
我尝试将这些元素存储到变量中,但没有插入任何内容到输出html中:
<xsl:variable name="e2Elements" select="//group[1]/e2"></xsl:variable>
<xsl:template match="//group">
<xsl:element name="group">
<e1>default1</e1>
<xsl:copy-of select="$e2Elements" />
</xsl:element>
</xsl:template>
但是我不确定如何将e2元素插入到值中。我使用的是SaxonHE9.8N,可以访问exslt命名空间和xslt2.0
发布于 2019-03-27 03:59:31
您的解决方案实际上是制作了一个不必要的元素副本。您可以在不复制它们的情况下完成此操作,如下所示:
<xsl:variable name="e3Elements" select="//group[1]/e2" />
另一个低效是使用前一个轴的<xsl:template match="group[preceding::group]"/>
总是很昂贵,尤其是在模式中。明显的改进是将其替换为前置兄弟(搜索前置兄弟轴比搜索前置同级轴快得多)。但实际上,您可以做得更好:将此规则设为组的默认规则(<xsl:template match="group"/>
),并使另一个规则仅与第一个组(<xsl:template match="group[1]">...
)匹配。
但实际上,也不需要匹配第一个group元素,因为您没有使用它的任何数据。
在风格上,<group>
比<xsl:element name="group">
更受欢迎,因为它的可读性更好。
这将是我的XSLT 3.0解决方案:
<xsl:transform version="3.0" .... expand-text="yes">
<xsl:template match="/">
<xsl:variable name="e3Elements" select="//group[1]/e2"/>
<xsl:for-each select="'default1', 'default2'">
<group>
<e1>{.}</e1>
<xsl:copy-of select="$e3Elements"/>
</group>
</xsl:for-each>
</xsl:template>
</xsl:transform>
发布于 2019-03-27 03:32:20
最后,我需要使用copy-of使我的变量成为元素的副本。下面是我的解决方案:
<xsl:template match="node() | @*">
<xsl:copy>
<xsl:apply-templates select="node() | @*"/>
</xsl:copy>
</xsl:template>
<xsl:variable name="e3Elements">
<xsl:copy-of select="//group[1]/e2" />
</xsl:variable>
<xsl:template match="group[preceding::group]"></xsl:template>
<xsl:template match="group">
<xsl:element name="group">
<xsl:element name="e1">default1</xsl:element>
<xsl:copy-of select="$e3Elements"></xsl:copy-of>
</xsl:element>
<xsl:element name="group">
<xsl:element name="e1">default2</xsl:element>
<xsl:copy-of select="$e3Elements"></xsl:copy-of>
</xsl:element>
</xsl:template>
https://stackoverflow.com/questions/55363510
复制相似问题