3

我想根据基于子元素的派生值对元素进行排序。派生值不能使用 XPath( 、 等)计算sumconcat但可以使用 XSL(xsl:choosexsl:if等)计算。

我会使用 EXSLT 功能扩展,但它不可用。环境是 XSLT 1.0、Xalan-C++ 版本 1.10,带有 EXSLT 通用和集合扩展。

编辑更改了示例以强调无法使用xsl:sort语句中的简单节点/xpath 函数计算我需要分组的派生值。

我的目标是在非活动药物之前列出当前药物,按开始日期降序排列。确定药物是否为最新的逻辑取决于药物是否已取消、尚未过期以及其他一些业务逻辑。

鉴于此 XML:

<?xml version="1.0"?>
<medications>
  <medication>
    <name>med1</name>
    <status>canceled</status>
    <startTime>2012-02-01T00:00:00Z</startTime>
    <endTime>2012-12-31T00:00:00Z</endTime>
    <!-- other elements omitted -->
  </medication>
  <medication>
    <name>med2</name>
    <status />
    <startTime>2012-01-01T00:00:00Z</startTime>
    <endTime>2012-01-07T00:00:00Z</endTime>
    <!-- other elements omitted -->
  </medication>
  <medication>
    <name>med3</name>
    <status />
    <startTime>2012-01-01T00:00:00Z</startTime>
    <!-- other element omitted -->
  </medication>
</medications>

样式表将生成一个排序的药物列表,包括示例数据中省略的信息(订购医生、药房位置等)和来自父节点的数据(患者地址、初级保健医生等)。对于此示例,我将生成一个简单的排序列表,显示可以遍历药物节点:

<?xml version="1.0" encoding="utf-8"?>
<medications>
  <medication isCurrent="1" name="med3" />
  <medication isCurrent="0" name="med1" />
  <medication isCurrent="0" name="med2" />
</medications>

我能想到的最好方法是将派生值预先计算到 EXSLT 节点集(以及排序所需的其他值),并使用键通过 generate-id 查找药物元素:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0"
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:exsl="http://exslt.org/common"
  exclude-result-prefixes="exsl">
  <xsl:output encoding="utf-8" method="xml" indent="yes" />

  <xsl:key name="medications-by-id" match="medication" use="generate-id()" />
  <xsl:variable name="medication-sorter">
    <xsl:for-each select="//medication">
      <item id="{generate-id(.)}">
        <xsl:attribute name="isCurrent">
          <xsl:apply-templates mode="isCurrentMedication" select="." />
        </xsl:attribute>
        <xsl:attribute name="startTime">
          <xsl:value-of select="startTime/text()" />
        </xsl:attribute>
      </item>
    </xsl:for-each>
  </xsl:variable>

  <xsl:template match="medications">
    <!-- hardcoded key lookup works -->
    <hardcoded><xsl:value-of select="key('medications-by-id',generate-id(medication[2]))/name/text()"/></hardcoded>

    <!-- but key lookup from the sort helper does not -->
    <medications>
      <xsl:for-each select="exsl:node-set($medication-sorter)/item">
        <xsl:sort select="@isCurrent" order="descending" />
        <xsl:sort select="@startTime" order="descending" />
        <medication>
          <xsl:attribute name="isCurrent">
            <xsl:value-of select="@isCurrent" />
          </xsl:attribute>
          <xsl:attribute name="name">
            <xsl:value-of select="key('medications-by-id',@id)/name/text()" />
          </xsl:attribute>
        </medication>
      </xsl:for-each>
    </medications>
  </xsl:template>

  <xsl:template mode="isCurrentMedication" match="medication">
    <xsl:choose>
      <xsl:when test="(status/text()='canceled') or (status/text()='discontinued') or (status/text()='inactive')">0</xsl:when>
      <xsl:otherwise>
        <!-- omitted date checking logic not relevent to this question, so just hardcoded -->
        <xsl:choose>
          <xsl:when test="name/text()='med2'">0</xsl:when>
          <xsl:when test="name/text()='med3'">1</xsl:when>
        </xsl:choose>
      </xsl:otherwise>
    </xsl:choose>
  </xsl:template>
</xsl:stylesheet>

然而,这并没有按预期工作。使用 generate-id(medication[2]) 查找密钥时,节点有效并且名称被输出,但在使用节点集中的 @id 调用时不起作用,即使值看起来完全相同:

<?xml version="1.0" encoding="utf-8"?>
<medications>
  <hardcoded>med2</hardcoded>
  <medication isCurrent="1" name="" />
  <medication isCurrent="0" name="" />
  <medication isCurrent="0" name="" />
</medications>

还使用 Xalan for Java 2.7.1 对此进行了测试,结果相同。

我可以通过copy-of在 $medication-sorter 节点集中包含药物元素来解决这个问题,但是父上下文丢失了,有时我的样式表需要它。

是否有另一种方法可以对必须使用 a 计算的值进行排序/分组xsl:template

4

2 回答 2

2

解决方案可能要简单得多:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output encoding="utf-8" method="xml" indent="yes"/>
<xsl:template match="books">
    <xsl:copy>
        <xsl:for-each select="book">
            <xsl:sort select="status" order="descending"/>
            <xsl:sort select="number(count) &gt; 0" order="descending"/>
            <xsl:copy-of select="."/>
        </xsl:for-each>
    </xsl:copy>
</xsl:template>
</xsl:stylesheet>
于 2012-04-18T06:33:19.933 回答
0

这一切看起来都太复杂了。仅排序而不分组或忽略重复项时,您不需要密钥。
试试下面的方法,它按状态和计数排序,两者都降序排列,以便在 Published 之后出现 Out of Print,并且在每个状态中最后出现零计数。

<?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" version="1.0" encoding="UTF-8" indent="yes"/>

    <xsl:template match="/">
        <output>
            <xsl:for-each select="books/book">
                <xsl:sort select="status" order="descending"/>
                <xsl:sort select="count" order="descending"/>
                <xsl:copy-of select="."/>
            </xsl:for-each>
        </output>
    </xsl:template>

</xsl:stylesheet>

结果与您指定的略有不同;要么你对此感到满意,要么我们需要做一些复杂化。

<?xml version="1.0" encoding="UTF-8"?>
<output>
    <book>
        <title>Three</title>
        <price>30</price>
        <count>3</count>
        <status>Published</status>
    </book>
    <book>
        <title>One</title>
        <price>10</price>
        <count>1</count>
        <status>Published</status>
    </book>
    <book>
        <title>Zero</title>
        <price>5</price>
        <count>0</count>
        <status>Published</status>
    </book>
    <book>
        <title>Two</title>
        <price/>
        <count>0</count>
        <status>Out of Print</status>
    </book>
</output>
于 2012-04-18T06:43:28.833 回答