gpt4 book ai didi

XSLT 排序使用派生/计算值而不使用 EXSLT 函数

转载 作者:行者123 更新时间:2023-12-02 02:16:42 24 4
gpt4 key购买 nike

我想根据基于子元素的派生值对元素进行排序。派生值不能使用 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>

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

<?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 节点集中(连同排序所需的其他值),并使用键通过生成 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 对此进行了测试,结果相同。

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

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

最佳答案

解决方案可能更简单:

<?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>

关于XSLT 排序使用派生/计算值而不使用 EXSLT 函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10203245/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com