2

这是一个非常复杂的问题,超出了我对 XSLT 的了解——我仍在学习,无论我读了多少 O'Reilly 的 XSLT 书,我都在我的头脑中。

我有一个多方面的问题,我已经为它生成了一个输入 XML 文件,之后我也会尝试解释这些要求。

输入

<roottag>
<body>
    <header>
        <r>
            <c>
                <d>Header Tag</d><!-- This can include spaces-->
                <e>System generated trash</e>
            </c>
        </r>
        <r>
            <c>
                <d>Sub Header Tag A</d>
                <e>System generated trash</e>
            </c>
            <c>
                <d>Sub Header Value A</d>
                <e>System generated trash</e>
            </c>
        </r>
        <r>
            <c>
                <d>Sub Header Tag B</d>
                <e>System generated trash</e>
            </c>
            <c>
                <d>Sub Header Value B</d>
                <e>System generated trash</e>
            </c>
        </r>
        <r>
            <c>
                <d>Sub Header Tag C</d>
                <e>System generated trash</e>
            </c>
            <c>
                <d>Sub Header Value C</d>
                <e>System generated trash</e>
            </c>
        </r>
    </header>
    <information>
        <r>Body of document</r>
        <r>Appears here but have an XSLT that deals with this</r>
    </informtaion>
    <footer>
        <r>
            <c>
                <d>Footer Tag</d><!-- This can include spaces-->
                <e>System generated trash</e>
            </c>
        </r>
        <r>
            <c>
                <d>Sub Footer Tag A</d>
                <e>System generated trash</e>
            </c>
            <c>
                <d>Sub Footer Value A</d>
                <e>System generated trash</e>
            </c>
        </r>
        <r>
            <c>
                <d>Sub Footer Tag B</d>
                <e>System generated trash</e>
            </c>
            <c>
                <d>Sub Footer Value B</d>
                <e>System generated trash</e>
            </c>
        </r>
        <r>
            <c>
                <d>Sub Footer Tag C</d>
                <e>System generated trash</e>
            </c>
            <c>
                <d>Sub Footer Value C</d>
                <e>System generated trash</e>
            </c>
        </r>
    </footer>
</body>
</roottag>

输出

<?xml version="1.0" encoding="utf-8"?>
<roottag>
  <body>
    <header>
      <HeaderTag>
        <!-- without spaces -->
        <HeaderName>Header Tag</HeaderName>
        <!-- This needs to preserve spaces-->
      </HeaderTag>
      <SubHeaderTagA>
        <!-- without spaces -->
        <HeaderName>Sub Header Tag A</HeaderName>
        <!-- This needs to preserve spaces-->
        <HeaderValue>Sub Header Value A</HeaderValue>
      </SubHeaderTagA>
      <SubHeaderTagB>
        <HeaderName>Sub Header Tag B</HeaderName>
        <HeaderValue>Sub Header Value B</HeaderValue>
      </SubHeaderTagB>
      <SubHeaderTagC>
        <HeaderName>Sub Header Tag C</HeaderName>
        <HeaderValue>Sub Header Value C</HeaderValue>
      </SubHeaderTagC>
    </header>
    <information>
      <r>Body of document</r>
      <r>Appears here but have an XSLT that deals with this</r>
      </information>
      <footer>
        <FooterTag>
          <FooterName>Footer Tag</FooterName>
        </FooterTag>
        <SubFooterTagA>
          <FooterName>Sub Footer Tag A</FooterName>
          <FooterValue>Sub Footer Value A</FooterValue>
        </SubFooterTagA>
        <SubFooterTagB>
          <FooterName>Sub Footer Tag B</FooterName>
          <FooterValue>Sub Footer Value B</FooterValue>
        </SubFooterTagB>
        <SubFooterTagC>
          <FooterName>Sub Footer Tag C</FooterName>
          <FooterValue>Sub Footer Value C</FooterValue>
        </SubFooterTagC>
      </footer>
    </body>
</roottag>

所以解释一下我看到的问题,以及我遇到的问题。

  1. RemovingSpaces: roottag/body/header/r/c/d 中保存的值可以并且通常包含空格,所以我需要一种方法来删除我从网站 [addLink] 上的问题中找到的这个值,但这也替换了值因此,当我稍后在该过程中使用数据时,它没有输出所需的空格。
  2. 仅用第一个值替换 R: 我不知道如何做到这一点,我尝试和研究的所有内容似乎都使用第二个值作为值。我已经结束了。
  3. 页眉或页脚名称/值: 同样,我的知识有限,如果这是可能的,或者每个标签都需要单独匹配?

  4. 页眉和页脚标签的移动: 我没有将其包含在所需的输出中,但我认为我可能需要这样做 - 是否可以将页眉标签和页脚标签移到正文标签之外?所以 XML 将是:roottag-header-body-information-/body-footer/-/rt

如果您需要更多说明,请告诉我。

4

2 回答 2

2

从源中的元素内容派生元素名称通常不是一个好主意 - 尽管您可以删除空格,但总是有可能出现其他特殊字符,即使您确实将它们全部删除,您最终也可能会出现意外重复。1 Tag例如,包含和的两个元素2 Tag都需要被剥离为Tag.

但是,这样的事情应该可以完成:

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output method="xml" indent="yes"/>
  <xsl:strip-space elements="*" />

  <xsl:variable name="allowed">ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvxyz_</xsl:variable>

  <xsl:template match="r[c/d]">
    <xsl:variable name="elemName" select="translate(c/d,translate(c/d,$allowed,''),'')" />
    <xsl:element name="{$elemName}">
      <xsl:apply-templates />
    </xsl:element>
  </xsl:template>

  <xsl:template match="c">
    <xsl:apply-templates select="@* | node()"/>
  </xsl:template>

  <xsl:template match="c[1]/d">
    <HeaderName>
      <xsl:apply-templates />
    </HeaderName>
  </xsl:template>

  <xsl:template match="c[2]/d">
    <HeaderValue>
      <xsl:apply-templates />
    </HeaderValue>
  </xsl:template>

  <xsl:template match="e" />

  <xsl:template match="@* | node()">
    <xsl:copy>
      <xsl:apply-templates select="@* | node()"/>
    </xsl:copy>
  </xsl:template>
</xsl:stylesheet>

这使用“双重翻译”方法从字符串中去除所有不需要的字符。该translate函数可用于从列表中删除所有字符,方法是指定一个空字符串来将这些字符转换为。您可以使用它从字符串中删除所有 VALID 字符,留下仅包含 INVALID 字符的字符串。然后,您再次使用 translate 从原始字符串中删除所有这些 INVALID 字符。

如果您确实需要正文之外的页眉/页脚,请添加以下模板:

<xsl:template match="roottag">
  <xsl:copy>
    <xsl:apply-templates select="body/header" />
    <xsl:apply-templates select="body" />
    <xsl:apply-templates select="body/footer" />
  </xsl:copy>
</xsl:template>

<xsl:template match="body">
  <xsl:copy>
    <xsl:apply-templates select="information" />
  </xsl:copy>
</xsl:template>

要忽略包含等号的r节点,请将此模板添加到匹配“ ”的下方:c\dr[c/d]

<xsl:template match="r[contains(c/d,'=')]" />
于 2012-12-10T13:34:01.193 回答
1

那么有多个问题!这么多答案..我会尽量系统地写,希望你能理解!并随时提出问题!

1.第一步是去掉标签之间额外的不需要的空格。为了实现这一点,在 XML 声明下面有以下两个语句作为标题:

  <xsl:output method="xml" indent="yes"/>
  <xsl:strip-space elements="*"/>

2.下一步是定义身份模板!它的工作是按原样从输入输出(命令不输出的除外)

  <xsl:template match="@* | node()">
    <xsl:copy>
      <xsl:apply-templates select="@* | node()"/>
    </xsl:copy>
  </xsl:template>

3.在你的输出中你不想看到<r>元素!所以我们将有以下几行

  <xsl:template match="/roottag/body/header/r
                      |/roottag/body/footer/r">
    <xsl:apply-templates select="@*|node()"/>
  </xsl:template>

这不会复制<r>元素,而是让子元素接管!我将在下面写..

4.您只想复制<c>每个元素中的第一个标签<r>..并忽略输出中的其余<c>节点..所以在模板覆盖下面定义..

  <xsl:template match="/roottag/body/header/r/c
                      |/roottag/body/footer/r/c"/>

这实际上会<c>从输出中删除所有元素!但下面的覆盖将<c>只处理第一个元素!

5.此代码用于<Header>在输出中构建元素!

  <xsl:template match="/roottag/body/header/r/c[1]">
    <xsl:variable name="ElementName">
      <xsl:call-template name="RemoveSpaceFromValueD">
        <xsl:with-param name="text" select="d/."/>
        <xsl:with-param name="replace" select="' '"/>
      </xsl:call-template>
    </xsl:variable>

    <xsl:element name="{$ElementName}">
      <xsl:element name="HeaderName">
        <xsl:value-of select="d/."/>
      </xsl:element>
      <xsl:if test="ancestor::r != ancestor::header/r[1]">
      <xsl:element name="HeaderValue">
        <xsl:value-of select="e/."/>
      </xsl:element>
      </xsl:if>
    </xsl:element>
  </xsl:template>

通过使用此代码,您不必为不同的模板声明不同的模板<header>, <footer> and <subHeaderTag>s

我在这里做的是:

  1. 调用一个新模板RemoveSpaceFromValueD,该模板将从元素中复制文本,从d文本中删除空格并将其分配给变量ElementName
  2. 使用来自变量的值ElementName将用于创建一个新元素!所以Sub Header Tag C将有<SubHeaderTagC>作为父母:)
  3. 将值复制<d><HeaderName>!
  4. <d>从下一个复制下一个<c><HeaderValue>

6. 也重复相同的代码<Footer>

  <xsl:template match="/roottag/body/footer/r/c[1]">
    <xsl:variable name="ElementName">
      <xsl:call-template name="RemoveSpaceFromValueD">
        <xsl:with-param name="text" select="d/."/>
        <xsl:with-param name="replace" select="' '"/>
      </xsl:call-template>
    </xsl:variable>

    <xsl:element name="{$ElementName}">
      <xsl:element name="FooterName">
        <xsl:value-of select="d/."/>
      </xsl:element>
      <xsl:if test="ancestor::r != ancestor::footer/r[1]">
        <xsl:element name="FooterValue">
          <xsl:value-of select="e/."/>
        </xsl:element>
      </xsl:if>
    </xsl:element>

  </xsl:template>

7.和模板RemoveSpaceFromValueD

  <xsl:template name="RemoveSpaceFromValueD">
    <xsl:param name="text"/>
    <xsl:param name="replace"/>
    <xsl:choose>
      <xsl:when test="contains($text,$replace)">
        <xsl:value-of select="substring-before($text,$replace)"/>
        <xsl:call-template name="RemoveSpaceFromValueD">
          <xsl:with-param name="text" select="substring-after($text,$replace)"/>
          <xsl:with-param name="replace" select="$replace"/>
        </xsl:call-template>
      </xsl:when>
      <xsl:otherwise>
        <xsl:value-of select="$text"/>
      </xsl:otherwise>
    </xsl:choose>
  </xsl:template>

8.整个 XSLT 代码如下所示:

测试这个!如果您有任何问题或理解困难,请告诉我。

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output method="xml" indent="yes"/>
  <xsl:strip-space elements="*"/>


  <xsl:template match="@* | node()">
    <xsl:copy>
      <xsl:apply-templates select="@* | node()"/>
    </xsl:copy>
  </xsl:template>


  <xsl:template match="/roottag/body/header/r
                      |/roottag/body/footer/r">
    <xsl:apply-templates select="@*|node()"/>
  </xsl:template>


  <xsl:template match="/roottag/body/header/r/c
                      |/roottag/body/footer/r/c"/>


  <xsl:template match="/roottag/body/header/r/c[1]">
    <xsl:variable name="ElementName">
      <xsl:call-template name="RemoveSpaceFromValueD">
        <xsl:with-param name="text" select="d/."/>
        <xsl:with-param name="replace" select="' '"/>
      </xsl:call-template>
    </xsl:variable>

    <xsl:element name="{$ElementName}">
      <xsl:element name="HeaderName">
        <xsl:value-of select="d/."/>
      </xsl:element>
      <xsl:if test="ancestor::r != ancestor::header/r[1]">
        <xsl:for-each select="../c[2]">
          <xsl:element name="HeaderValue">
            <xsl:value-of select="d/."/>
          </xsl:element>
        </xsl:for-each>
      </xsl:if>
    </xsl:element>

  </xsl:template>


  <xsl:template match="/roottag/body/footer/r/c[1]">
    <xsl:variable name="ElementName">
      <xsl:call-template name="RemoveSpaceFromValueD">
        <xsl:with-param name="text" select="d/."/>
        <xsl:with-param name="replace" select="' '"/>
      </xsl:call-template>
    </xsl:variable>

    <xsl:element name="{$ElementName}">
      <xsl:element name="FooterName">
        <xsl:value-of select="d/."/>
      </xsl:element>
      <xsl:for-each select="../c[2]">
        <xsl:element name="FooterValue">
          <xsl:value-of select="d/."/>
        </xsl:element>
      </xsl:for-each>
    </xsl:element>

  </xsl:template>


  <xsl:template name="RemoveSpaceFromValueD">
    <xsl:param name="text"/>
    <xsl:param name="replace"/>
    <xsl:choose>
      <xsl:when test="contains($text,$replace)">
        <xsl:value-of select="substring-before($text,$replace)"/>
        <xsl:call-template name="RemoveSpaceFromValueD">
          <xsl:with-param name="text" select="substring-after($text,$replace)"/>
          <xsl:with-param name="replace" select="$replace"/>
        </xsl:call-template>
      </xsl:when>
      <xsl:otherwise>
        <xsl:value-of select="$text"/>
      </xsl:otherwise>
    </xsl:choose>
  </xsl:template>


</xsl:stylesheet>
于 2012-12-10T13:18:10.720 回答