0

我的问题与类似,作者询问如何找到一组元素的最小属性。我是 xpath 的初学者,我无法接受该答案并将其添加到我的 XSLT。我需要找到一个计划的最低价格(见下文)并向包含该最低价格的细分添加一个属性。

我有以下 XML 片段:

<Subdivision Status="Active">
  <SubdivisionName>Chesla</SubdivisionName>
  <Plan Type="SingleFamily">
    <PlanNumber>555</PlanNumber>
    <PlanName>Pinewood</PlanName>
    <BasePrice>200500.0000</BasePrice>
  </Plan>
  <Plan Type="MultiFamily">
    <PlanNumber>444</PlanNumber>
    <PlanName>Westfield</PlanName>
    <BasePrice>270000.0000</BasePrice>
  </Plan>

注意:我的 XML 文档中可以有多个细分。为简洁起见,我只是在示例中显示一个。

我的最终输出与输入相同,只是在 Subdivision 元素中添加了 PriceLow 属性,该元素包含作为 Subdivision 兄弟的所有 Plans 的最小 BasePrice:

<Subdivision Status="Active" PriceLow="200500.00"> <==== ADDED!!!
  <SubdivisionName>Chesla</SubdivisionName>
  <Plan Type="SingleFamily">
    <PlanNumber>555</PlanNumber>
    <PlanName>Pinewood</PlanName>
    <BasePrice>200500.00</BasePrice>
  </Plan>
  <Plan Type="MultiFamily">
    <PlanNumber>444</PlanNumber>
    <PlanName>Westfield</PlanName>
    <BasePrice>270000.00</BasePrice>
  </Plan>

我不完整的 xsl 文件如下所示:

<xsl:output omit-xml-declaration="yes" indent="yes"/>

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

<xsl:template match="Subdivision">
   ... something goes here ...
</xsl:template>

有人可以帮我完成这个 XSLT 文件吗?我正在阅读一些关于 XSLT 的书籍,但与此同时我需要尽快完成这个项目。

干杯,布雷特

4

1 回答 1

0

这应该这样做:

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

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

  <xsl:template match="Subdivision">
    <xsl:copy>
      <xsl:apply-templates select="@*" />
      <xsl:attribute name="PriceLow">
        <xsl:apply-templates select="Plan/BasePrice" mode="min">
          <xsl:sort select="." order="ascending" data-type="number" />
        </xsl:apply-templates>
      </xsl:attribute>

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

  <xsl:template match="node()" mode="min">
    <xsl:if test="position() = 1">
      <xsl:value-of select="."/>
    </xsl:if>
  </xsl:template>
</xsl:stylesheet>

或者更简洁但可能效率较低的方法:

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

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

  <xsl:template match="Subdivision">
    <Subdivision PriceLow="{Plan/BasePrice[not(. > current()/Plan/BasePrice)]}">
      <xsl:apply-templates select="@*[name() != 'PriceLow'] | node()" />
    </Subdivision>
  </xsl:template>
</xsl:stylesheet>

在此示例数据上运行其中任何一个时:

<root>
  <Subdivision Status="Active">
    <SubdivisionName>Chesla</SubdivisionName>
    <Plan Type="SingleFamily">
      <PlanNumber>555</PlanNumber>
      <PlanName>Pinewood</PlanName>
      <BasePrice>200500.0000</BasePrice>
    </Plan>
    <Plan Type="MultiFamily">
      <PlanNumber>444</PlanNumber>
      <PlanName>Westfield</PlanName>
      <BasePrice>270000.0000</BasePrice>
    </Plan>
  </Subdivision>

  <Subdivision Status="Active">
    <SubdivisionName>Newhaven</SubdivisionName>
    <Plan Type="SingleFamily">
      <PlanNumber>555</PlanNumber>
      <PlanName>Pinewood</PlanName>
      <BasePrice>200900.0000</BasePrice>
    </Plan>
    <Plan Type="MultiFamily">
      <PlanNumber>444</PlanNumber>
      <PlanName>Westfield</PlanName>
      <BasePrice>200100.0000</BasePrice>
    </Plan>
    <Plan Type="QuadrupleFamily">
      <PlanNumber>111</PlanNumber>
      <PlanName>Pinewood</PlanName>
      <BasePrice>200050.0000</BasePrice>
    </Plan>
    <Plan Type="QuintupleFamily">
      <PlanNumber>222</PlanNumber>
      <PlanName>Pinewood</PlanName>
      <BasePrice>200300.0000</BasePrice>
    </Plan>
  </Subdivision>
</root>

结果是:

<root>
  <Subdivision Status="Active" PriceLow="200500.0000">
    <SubdivisionName>Chesla</SubdivisionName>
    <Plan Type="SingleFamily">
      <PlanNumber>555</PlanNumber>
      <PlanName>Pinewood</PlanName>
      <BasePrice>200500.0000</BasePrice>
    </Plan>
    <Plan Type="MultiFamily">
      <PlanNumber>444</PlanNumber>
      <PlanName>Westfield</PlanName>
      <BasePrice>270000.0000</BasePrice>
    </Plan>
  </Subdivision>

  <Subdivision Status="Active" PriceLow="200050.0000">
    <SubdivisionName>Newhaven</SubdivisionName>
    <Plan Type="SingleFamily">
      <PlanNumber>555</PlanNumber>
      <PlanName>Pinewood</PlanName>
      <BasePrice>200900.0000</BasePrice>
    </Plan>
    <Plan Type="MultiFamily">
      <PlanNumber>444</PlanNumber>
      <PlanName>Westfield</PlanName>
      <BasePrice>200100.0000</BasePrice>
    </Plan>
    <Plan Type="QuadrupleFamily">
      <PlanNumber>111</PlanNumber>
      <PlanName>Pinewood</PlanName>
      <BasePrice>200050.0000</BasePrice>
    </Plan>
    <Plan Type="QuintupleFamily">
      <PlanNumber>222</PlanNumber>
      <PlanName>Pinewood</PlanName>
      <BasePrice>200300.0000</BasePrice>
    </Plan>
  </Subdivision>
</root>
于 2013-04-15T22:17:23.053 回答