我有多个元素包含一个uniqueId(生成并存储在变量中)。我已经递归地添加了元素(对象类=具有唯一 id 的捆绑包)与 xsl
如下<xsl:template match="visualChildren">
<object class="Set" >
<installChildren>
<xsl:call-template name="Bundle">
<xsl:with-param name="i" select="1"/>
<xsl:with-param name="limit" select="4" />
</xsl:call-template>
</installChildren>
</object>
</xsl:template>
<xsl:template name="Bundle">
<xsl:param name="i"/>
<xsl:param name="limit"/>
<xsl:variable name="BundleObjId">
<xsl:value-of select="php:function('GenerateObjId')"/>
</xsl:variable>
<xsl:if test="$i <= $limit">
<object class="Bundle" objectID="{$BundleObjId}">
<property></property>
</object>
</xsl:if>
<xsl:call-template name="Bundle">
<xsl:with-param name="i" select="$i+1"/>
<xsl:with-param name="limit" select="$limit" />
</xsl:call-template>
</xsl:template>
这将产生以下结果
<visualChildren>
<object class="Set" >
<installChildren>
<object class="Bundle" objectID="33110emc908m">
<property></property>
</object>
<object class="Bundle" objectID="43110emc9667m">
<property></property>
</object>
</installChildren>
</object>
</visualChildren>
现在我需要将 BundleObjId 填充为父级的同级,以便引用它。
所需输出为:
<visualChildren>
<object class="Set" >
<installChildren>
<object class="Bundle" objectID="33110emc908m">
<property></property>
</object>
<object class="Bundle" objectID="43110emc9667m">
<property></property>
</object>
</installChildren>
</object>
<object RefId=33110emc908m />
<object RefId=43110emc9667m />
</visualChildren>
请通过添加到现有的 xsl,帮助我使用 xslt 1.0 实现这一目标。
如果不看到输入,很难提供答案。我相信您需要执行以下操作:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:exsl="http://exslt.org/common"
extension-element-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:template match="visualChildren">
<xsl:variable name="bundles">
<xsl:call-template name="create-bundles">
<xsl:with-param name="n" select="4" />
</xsl:call-template>
</xsl:variable>
<visualChildren>
<object class="Set" >
<installChildren>
<xsl:for-each select="exsl:node-set($bundles)/object">
<object class="Bundle" objectID="{@RefId}">
<property/>
</object>
</xsl:for-each>
</installChildren>
</object>
<xsl:copy-of select="$bundles"/>
</visualChildren>
</xsl:template>
<xsl:template name="create-bundles">
<xsl:param name="n"/>
<xsl:if test="$n > 0">
<object RefId="{php:function('GenerateObjId')}"/>
<xsl:call-template name="create-bundles">
<xsl:with-param name="n" select="$n - 1" />
</xsl:call-template>
</xsl:if>
</xsl:template>
</xsl:stylesheet>
给定以下人工输入:
<?xml version="1.0" encoding="UTF-8"?>
<visualChildren/>
结果将是这样的:
<?xml version="1.0" encoding="UTF-8"?>
<visualChildren>
<object class="Set">
<installChildren>
<object class="Bundle" objectID="123456">
<property/>
</object>
<object class="Bundle" objectID="987654">
<property/>
</object>
<object class="Bundle" objectID="456321">
<property/>
</object>
<object class="Bundle" objectID="789456">
<property/>
</object>
</installChildren>
</object>
<object RefId="123456"/>
<object RefId="987654"/>
<object RefId="456321"/>
<object RefId="789456"/>
</visualChildren>
问题是你从 1 迭代到 N,而不是迭代节点集。 在该迭代中,您调用一个返回值不依赖于上下文的函数(显然,它正在生成一个随机字符串)。 因此,您没有任何机会在 2 次不同的迭代中重复相同的 ID。 在 XSLT 1.0 中,不能生成 ID 的中间列表,然后循环访问该列表。 我看到 3 个解决方案:
-
如果
php:function('GenerateObjId')
可以用generate-id()
替换,则迭代节点,并使用后者(它将为 2 个相同的节点生成相同的 ID) -
如果您的处理器支持
exsl:node-set()
(大多数都支持,此处的文档),请生成所需的 ID 列表,作为包含单个父元素的简单 XML 元素列表,将其存储在变量中,然后使用exsl:node-set()
迭代(两次) -
如果可以,请切换到 XSLT 2.0(您可以在其中应用以前的解决方案,而无需
exsl:node-set()
,因为在 XSLT 2.0 中,您可以使用 XSLT 1.0 中的结果树作为 XPath 表达式的输入)
最后一个"解决方案"(如果以上都不可能):
- 根据修改后的身份转换模式,对第一个转换的结果应用第二个转换:
'' <xsl:样式表>
<xsl:template match="node()">
<xsl:copy>
<xsl:copy-of select="@*"/>
<xsl:apply-templates select="node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="object[@class = 'Set']">
<xsl:copy-of select="."/>
<xsl:for-each select="installChildren/object">
<object RefId="{ @objectID }"/>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
'''