【问题标题】:transform siblings nodes to imbricated nodes将兄弟节点转换为叠瓦状节点
【发布时间】:2011-08-05 12:37:11
【问题描述】:

我遇到了 XSLT 问题。我在<source> 中有一组兄弟节点(<level>),我想将其转换为节点的叠层(即每个级别都将呈现在其前一个兄弟节点中)。

XML 输入

<?xml version="1.0"?>
<sources> 
    <source mode="manual"  name="test1">
                <level>blablabla Level1</level>
                <level>this is the second level</level>
                <level>this is the third level</level>
    </source>
</sources>

预期输出

我想要的输出是这个的叠印 html 版本(删节,叠印就是这里的东西):

   <form class="source manual">
    source &gt; <input value="test1" name="sourceName" type="text">

    <!-- LEVEL #1 -->
    <p class="deepnessIndicator">Deepness: <strong>1</strong></p>
    <div class="deepnessContainer">
         <!-- LEVEL #2 -->
         next-level:
         <p class="deepnessIndicator">Deepness: <strong>2</strong></p>
         <div class="deepnessContainer">
              <!-- LEVEL #3 -->
              next-level:
              <p class="deepnessIndicator">Deepness: <strong>3</strong></p>
         </div>
    </div>
  </form>

不幸的是我编写的 XSL 失败了,这里是源代码(我试图缩短但是):

XSL

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

    <xsl:template match="sources">
        <xsl:apply-templates select="*" />
    </xsl:template>

    <!-- Main template here -->
    <xsl:template match="source[@mode='manual']">
        <form class="source manual">
            source &gt; <input type="text" name="sourceName" value="{@name}" />

            <!-- Here's what I call first the recursion 
                the parameter is the # of the <level> 
                that should be processed -->
            <xsl:call-template name="sourceLevelRecursion">
                <xsl:with-param name="currentLevel">1</xsl:with-param>
            </xsl:call-template>

        </form>
    </xsl:template>

    <!-- Recursion template -->
    <xsl:template name="sourceLevelRecursion">
        <xsl:param name="currentLevel" />

            <!-- this apply-templates should apply on only 
                one node because of the selector but it won't -->
        <xsl:apply-templates mode="deepnessHeader" select="./level[$currentLevel]">
            <xsl:with-param name="currentLevel"><xsl:value-of select="$currentLevel" /></xsl:with-param>
        </xsl:apply-templates>

        <xsl:if test="level[$currentLevel+1]">
            <div class="deepnessContainer">
                    <!-- Recursion Call here -->
            <xsl:call-template name="sourceLevelRecursion">
                <xsl:with-param name="currentLevel"><xsl:value-of select="$currentLevel+1" /></xsl:with-param>
                </xsl:call-template>
            </div>
                </xsl:if>
    </xsl:template>

        <xsl:template mode="deepnessHeader" match="level">
        <xsl:param name="currentLevel" />
            <p class="deepnessIndicator">Deepness: <strong><xsl:value-of select="$currentLevel" /></strong></p>
        </xsl:template>


<xsl:template match="text()" />

不幸的输出

我得到的最终错误输出是:

   <form class="source manual">
    source &gt; <input value="test1" name="sourceName" type="text">

    <p class="deepnessIndicator">Deepness: <strong>1</strong></p>
    <p class="deepnessIndicator">Deepness: <strong>1</strong></p>
    <p class="deepnessIndicator">Deepness: <strong>1</strong></p>

    <div class="deepnessContainer">
      next-level:

      <p class="deepnessIndicator">Deepness: <strong>2</strong></p>
      <p class="deepnessIndicator">Deepness: <strong>2</strong></p>
      <p class="deepnessIndicator">Deepness: <strong>2</strong></p>

      <div class="deepnessContainer">
        next-level:

        <p class="deepnessIndicator">Deepness: <strong>3</strong></p>
        <p class="deepnessIndicator">Deepness: <strong>3</strong></p>
        <p class="deepnessIndicator">Deepness: <strong>3</strong></p>
      </div>
    </div>
  </form>

如您所见,应用:

&lt;xsl:apply-templates mode="deepnessHeader" select="./level[$currentLevel]"&gt;

匹配 &lt;xsl:template mode="deepnessHeader" match="level"&gt;

匹配三次,源 XML 中每个 &lt;level&gt; 匹配一次。但是,apply-templates 中的选择器应该只选择一个节点不是吗?

【问题讨论】:

  • PS:我用这个运行了我的 XSLT 测试:xslttest.appspot.com
  • 好问题,+1。请参阅我的答案以获得完整且简单的解决方案以及此问题原因的解释。

标签: xslt


【解决方案1】:

使用

<xsl:apply-templates mode="deepnessHeader" 
   select="./level[position()=$currentLevel]">

XSLT 1.0 是“弱类型”。 XSLT 处理器不知道$currentLevel 包含的值应被视为整数

因此$currentLevel 被视为布尔值——谓词中的任何非整数表达式都应如此。但是,如果可以将实际值强制转换为整数,则任何不同于 0 的整数值都将被视为true(),整个谓词为true(),因此不会过滤掉任何内容。

记住

在 XPath 1.0 中,任何 Expr[someInteger] (其中 someInteger 是整数文字)是以下形式的简写:Expr[position() = someInteger]

【讨论】:

  • 非常感谢,完美运行!然后,我又问了两个问题:为什么测试 一开始就失败了?不应该永远是真的吗?
  • 第二个问题,您提到了 XSLT 1.0,那么它在 XSLT2 中有什么不同吗? (只是好奇,不能使用 xslt2 因为我在浏览器中执行转换)
  • @BiAiB:您的第一个问题:在level[$currentLevel+1] 中,XSLT 处理器知道谓词内的表达式是数字类型(因为加法运算符)。因此,它将其视为:level[position() = $currentLevel+1]
  • 这意味着&lt;xsl:apply-templates mode="deepnessHeader" select="./level[$currentLevel+0]"&gt; 也可以是一个修正?
  • @BiAiB:您的第二个问题:在 XPath 2.0/XSLT 2.0 中存在强类型,并且变量可能具有为自己声明的显式类型。例如,如果您有:&lt;xsl:variable name="currentLevel" select="someExpression" as="xs:integer"/&gt;,那么在 level[$currentLevel] 中,XSLT 处理器知道谓词内的表达式是 xs:integer 类型并扩展简写。
猜你喜欢
  • 2021-11-18
  • 1970-01-01
  • 1970-01-01
  • 2014-02-05
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多