虽然问题中没有提到,但我猜您的问题是您收到消息“Keyword xsl:sort may not be used here.”,这是因为您的xsl:sort嵌套在xsl:when应该是什么时候直接在xsl:for-each下
实际上,您正在尝试根据您的参数进行条件排序。一个条件是是否按 x 或 y 坐标排序,另一个条件是按升序还是降序排序。
对于排序,您可以定义一个变量,该变量将保存“升序”或“降序”,然后按以下顺序排序:
<xsl:variable name="order">
<xsl:choose>
<xsl:when test="$MinOrMax='max'">descending</xsl:when>
<xsl:otherwise>ascending</xsl:otherwise>
</xsl:choose>
</xsl:variable>
<xsl:for-each select="coordinates/point">
<xsl:sort select="substring-before(substring-after(.,'('),'/')" data-type="number" order="{$order}"/>
下一个问题是如何根据您的参数选择“x”或“y”坐标。一种方法是使用xsl:choose在排序方法之间进行选择
<xsl:choose>
<xsl:when test="$XorY='x'">
<xsl:for-each select="coordinates/point">
<xsl:sort select="substring-before(substring-after(.,'('),'/')" data-type="number" order="{$order}"/>
<xsl:if test="position() = 1">
<xsl:value-of select="substring-before(substring-after(.,'('),'/')"/>
</xsl:if>
</xsl:for-each>
</xsl:when>
<xsl:otherwise>
<xsl:for-each select="coordinates/point">
<xsl:sort select="substring-before(substring-after(.,'/'),')')" data-type="number" order="{$order}"/>
<xsl:if test="position() = 1">
<xsl:value-of select="substring-before(substring-after(.,'/'),')')"/>
</xsl:if>
</xsl:for-each>
</xsl:otherwise>
</xsl:choose>
但是,有一种方法可以避免这种潜在的代码重复,并且只有一个xsl:for-each。试试这个
<xsl:sort select="concat(
substring(substring-before(substring-after(.,'('),'/'), 1, 100 * ($XorY = 'x')),
substring(substring-before(substring-after(.,'/'),')'), 1, 100 * ($XorY = 'y')))"
data-type="number" order="{$order}"/>
这利用($XorY = 'x')
了表达式中将评估为 0 或 1 的事实。为了更好地理解它,它实际上是对这两个表达式进行串联:
substring(<x co-ordinate>, 1, 100 * <0 or 1>)
substring(<y co-ordinate>, 1, 100 * <0 or 1>)
当 $XorY = 'x' 时返回 1,因此第一个子字符串(x 坐标)返回一个完整字符串,但第二个子字符串返回一个空字符串。当 $XorY = 'y' 时,则相反,并返回第二个子字符串(y 坐标)。
试试这个 XSLT
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" indent="yes"/>
<xsl:template match="polygon">
<polygon>
<xMin>
<xsl:call-template name="getMinOf">
<xsl:with-param name="MinOrMax" select="'min'" />
<xsl:with-param name="XorY" select="'x'" />
</xsl:call-template>
</xMin>
<xMax>
<xsl:call-template name="getMinOf">
<xsl:with-param name="MinOrMax" select="'max'" />
<xsl:with-param name="XorY" select="'x'" />
</xsl:call-template>
</xMax>
<yMin>
<xsl:call-template name="getMinOf">
<xsl:with-param name="MinOrMax" select="'min'" />
<xsl:with-param name="XorY" select="'y'" />
</xsl:call-template>
</yMin>
<yMax>
<xsl:call-template name="getMinOf">
<xsl:with-param name="MinOrMax" select="'max'" />
<xsl:with-param name="XorY" select="'y'" />
</xsl:call-template>
</yMax>
</polygon>
</xsl:template>
<xsl:template name="getMinOf">
<xsl:param name="MinOrMax"/>
<xsl:param name="XorY"/>
<xsl:variable name="order">
<xsl:choose>
<xsl:when test="$MinOrMax='max'">descending</xsl:when>
<xsl:otherwise>ascending</xsl:otherwise>
</xsl:choose>
</xsl:variable>
<xsl:for-each select="coordinates/point">
<xsl:sort select="concat(substring(substring-before(substring-after(.,'('),'/'), 1, 100 * ($XorY = 'x')), substring(substring-before(substring-after(.,'/'),')'), 1, 100 * ($XorY = 'y')))" data-type="number" order="{$order}"/>
<xsl:if test="position() = 1">
<xsl:value-of select="concat(substring(substring-before(substring-after(.,'('),'/'), 1, 100 * ($XorY = 'x')), substring(substring-before(substring-after(.,'/'),')'), 1, 100 * ($XorY = 'y')))"/>
</xsl:if>
</xsl:for-each>
</xsl:template>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
应用于您的示例 XML 时,将输出以下内容
<polygon>
<xMin>2.456</xMin>
<xMax>8.234</xMax>
<yMin>5.678</yMin>
<yMax>9.435</yMax>
</polygon>