我想基于子元素基于派生值对元素进行排序。无法使用XPath(sum
,concat
等)计算派生值,但可以使用XSL(xsl:choose
,xsl: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(药物[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对此进行了测试,结果相同。
我可以通过在$ medication-sorter节点集中包含copy-of
药物元素来解决这个问题,但是父上下文会丢失,有时我的样式表会需要它。
是否有其他方法可以对必须使用xsl:template
计算的值进行排序/分组?
答案 0 :(得分: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) > 0" order="descending"/>
<xsl:copy-of select="."/>
</xsl:for-each>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
答案 1 :(得分:0)
这看起来太复杂了。只是排序而不是分组或忽略重复项时,您不需要密钥 尝试以下操作,对状态和计数进行排序,两者都降序,以便在发布后发生绝版,并且在每个状态中最后一次发生零计数。
<?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>