检索具有(或不具有)子节点的节点以应用条件模板

时间:2013-11-22 09:19:06

标签: xslt parameters selectnodes

我阅读了很多文章,但没有找到对我的问题的确凿帮助。

我有一个XML文档,我应用xslt来获取csv文件作为输出。 我向xsl转换发送一个参数来过滤目标节点以应用模板。

xml文档看起来像那样(我删除了一些用于理解的无用节点):

<GetMOTransactionsResponse xmlns="http://www.exane.com/pott" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.exane.com/pott PoTTMOTransaction.xsd">
<MOTransaction>
    <Transaction VersionNumber="2" TradeDate="2013-11-20">
        <TransactionId Type="Risque">32164597</TransactionId>
        <InternalTransaction Type="Switch">
            <BookCounterparty>
                <Id Type="Risque">94</Id>
            </BookCounterparty>
        </InternalTransaction>
        <SalesPerson>
            <Id Type="Risque">-1</Id>
        </SalesPerson>
    </Transaction>
    <GrossPrice>58.92</GrossPrice>
    <MOAccount Account="TO1E" />
    <Entity>0021</Entity>
</MOTransaction>
<MOTransaction>
    <Transaction VersionNumber="1" TradeDate="2013-11-20">
        <TransactionId Type="Risque">32164598</TransactionId>
        <SalesPerson>
            <Id Type="Risque">-1</Id>
        </SalesPerson>
    </Transaction>
    <GrossPrice>58.92</GrossPrice>
    <MOAccount Account="TO3E" />
    <Entity>0021</Entity>
</MOTransaction>
</GetMOTransactionsResponse>

我的xslt在下面(抱歉它很长,而且我写得比实际更简单):

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

  <xsl:output method="text" omit-xml-declaration="no" indent="no" />

  <xsl:param name="instrumentalSystem"></xsl:param>

  <xsl:template name="abs">
    <xsl:param name="n" />
    <xsl:choose>
      <xsl:when test="$n = 0">
        <xsl:text>0</xsl:text>
      </xsl:when>
      <xsl:when test="$n &gt; 0">
        <xsl:value-of select="format-number($n, '#')" />
      </xsl:when>
      <xsl:otherwise>
        <xsl:value-of select="format-number(0 - $n, '#')" />
      </xsl:otherwise>
    </xsl:choose>
  </xsl:template>
  <xsl:template name="outputFormat">
    <!--Declaration of variables-->
    <xsl:variable name="GrossPrice" select="pott:GrossPrice" />
    <xsl:variable name="TransactionId" select="pott:Transaction/pott:TransactionId[@Type='Risque']" />
    <xsl:variable name="VersionNumber" select="pott:Transaction/@VersionNumber" />

    <!--Set tags values-->
    <xsl:value-of select="$Entity" />
    <xsl:text>;</xsl:text>
    <xsl:value-of select="concat('0000000', pott:MOAccount/@Account) "/>
    <xsl:text>;</xsl:text>

    <xsl:text>;</xsl:text>
    <xsl:value-of select="$TransactionId" />
    <xsl:text>;</xsl:text>
    <xsl:value-of select="$VersionNumber" />
    <xsl:text>&#13;&#10;</xsl:text>
  </xsl:template>
  <xsl:template match="/">
    <xsl:choose>
      <!-- BB -->
      <xsl:when test="$instrumentalSystem = 'BB'">
        <!--xsl:for-each select="pott:GetMOTransactionsResponse/pott:MOTransaction/pott:Transaction[pott:InternalTransaction]"-->
        <xsl:for-each select="pott:GetMOTransactionsResponse/pott:MOTransaction/pott:Transaction[pott:InternalTransaction]">
          <xsl:call-template name="outputFormat"></xsl:call-template>
        </xsl:for-each>
      </xsl:when>

      <!-- CP -->
      <xsl:when test="$instrumentalSystem = 'CP'">
        <xsl:for-each select="pott:GetMOTransactionsResponse/pott:MOTransaction/pott:Transaction[not(pott:InternalTransaction)]">
          <xsl:call-template name="outputFormat"></xsl:call-template>
        </xsl:for-each>
      </xsl:when>

      <xsl:otherwise>
      </xsl:otherwise>
    </xsl:choose>

  </xsl:template>

</xsl:stylesheet>

如果参数= BB,我想选择具有包含 InternalTransaction 节点的子事务的 MOTransaction 节点。

如果参数= CP,我想选择 MOTransaction 节点,这些节点没有包含 InternalTransaction 节点的子事务

当我写作 pott:GetMOTransactionsResponse / pott:MOTransaction / pott:Transaction [pott:InternalTransaction],我获取Transaction节点而不是MOTransaction节点 我想我距离预期的结果并不是很远,但尽管我尝试了所有的尝试,但我失败了。 如果有人可以帮助我。

我希望明确,否则我可以提供更多信息。

1 个答案:

答案 0 :(得分:0)

查看 xsl:for-each 语句之一,您正在执行此操作

<xsl:for-each select="pott:GetMOTransactionsResponse/pott:MOTransaction/pott:Transaction[pott:InternalTransaction]">

您说要选择 MOTransaction 元素,但它实际上是选择子事务元素。为了匹配您需要的逻辑,应该是这个

<xsl:for-each select="pott:GetMOTransactionsResponse/pott:MOTransaction[pott:Transaction[pott:InternalTransaction]]">

事实上,这也应该有用

<xsl:for-each select="pott:GetMOTransactionsResponse/pott:MOTransaction[pott:Transaction/pott:InternalTransaction]">

同样,对于第二个语句(在参数为“CP”的情况下),它可能看起来像这样

<xsl:for-each select="pott:GetMOTransactionsResponse/pott:MOTransaction[pott:Transaction[not(pott:InternalTransaction)]]">

或者,它可能看起来像这样

<xsl:for-each select="pott:GetMOTransactionsResponse/pott:MOTransaction[not(pott:Transaction/pott:InternalTransaction)]">

它们并不完全相同,因为第一个只包含具有 Transaction 子元素的 MOTransaction 元素,而第二个将包含 MOTransaction 根本没有任何交易孩子。

稍微说,您不需要在此处使用 xsl:for-each xsl:call-template 。使用模板匹配可能更好。

首先,尝试将命名模板<xsl:template name="outputFormat">更改为此

<xsl:template match="pott:MOTransaction">

然后,您可以重新编写将 xsl:for-each xsl:call-template 合并为一个 xsl:apply-templates < / strong>致电。

<xsl:apply-template select="pott:GetMOTransactionsResponse/pott:MOTransaction[pott:Transaction/pott:InternalTransaction]" />