在 xslt 1.0 中向父元素添加子属性



我有多个元素包含一个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 &lt;= $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>
'

''

相关内容

  • 没有找到相关文章

最新更新