xml – XSLT函数返回不同的结果[Saxon-EE vs Saxon-HE / PE]

我目前正在使用各种版本的Saxon-Processor进行纯XSL转换.下面是我的简短样式表,简化了我的问题的需求:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="2.0"
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    xmlns:foo="bar">

    <xsl:output encoding="UTF-8" method="text"/>

    <xsl:template match="/">
        <xsl:text>Call of func_1: </xsl:text>        
        <xsl:value-of select="foo:func_1()"/>

        <xsl:text>&#xA;Call of func_1: </xsl:text>
        <xsl:value-of select="foo:func_1()"/>

        <xsl:text>&#xA;Call of func_1: </xsl:text>
        <xsl:value-of select="foo:func_1()"/>

        <xsl:text>&#xA;Call of func_2: </xsl:text>
        <xsl:value-of select="foo:func_2()"/>
    </xsl:template>

    <xsl:function name="foo:func_1" as="xs:string">
        <!-- do some other stuff -->
        <xsl:value-of select="foo:func_2()"/>
    </xsl:function>

    <xsl:function name="foo:func_2" as="xs:string">
        <xsl:variable name="node">
            <xsl:comment/>
        </xsl:variable>
        <xsl:sequence select="generate-id($node)"/>
    </xsl:function>

</xsl:stylesheet>

描述

foo:func_1是一个包装器函数,用于返回执行其他操作的第二个函数的值,可以忽略.这个函数的概念调用其他函数是强制性的!

foo:func_2为元素生成唯一的id.此元素在名为“node”的本地范围变量中创建.

基于Saxon版本的不同结果

预期结果:

Call of func_1: d2
Call of func_1: d3
Call of func_1: d4
Call of func_2: d5

Saxon-EE 9.6.0.7 / Saxon-EE 9.6.0.5结果

Call of func_1: d2
Call of func_1: d2
Call of func_1: d2
Call of func_2: d3

Saxon-HE 9.6.0.5 / Saxon-PE 9.6.0.5 / Saxon-EE 9.5.1.6 / Saxon-HE 9.5.1.6结果

like expected

问题/进一步深入

我尽我所能调试了这个问题.如果我将函数“func_1”中的xsl:value-of更改为xsl:sequence,则所有版本[如预期]的结果将相同.但那不是我的意图!

我想了解,在整个Saxon版本中,xsl:value-of和xsl:sequence之间有什么区别.
有没有“隐藏”的缓存?在我的情况下,使用xsl:sequence和xsl:value-of的正确方法是什么. [顺便说一句:我已经知道了,value-of创建了一个带有select语句结果的文本节点. sequence可以是对节点或原子值的引用.不解决我的问题afaik]

这是一个长期存在且相当深刻的问题.在纯函数式语言中,使用相同的参数调用两次纯函数总会产生相同的结果.这使得许多优化成为可能,例如,如果参数是不变的,则将函数调用从循环中拉出,或者如果函数调用不是递归的,则内联函数调用.不幸的是,XSLT和XQuery函数并不是非常纯粹的功能:特别是,它们被定义为如果函数创建新节点,则调用该函数两次会产生不同的节点(f()是f()返回false).

Saxon优化器在这些约束条件下尽可能地进行优化,特别是通过识别创建新节点的函数并避免对这些函数进行积极优化.

但规范本身并非100%规范.例如,如果在您的示例中有一个局部变量没有依赖于函数参数,我认为规范为实现提供了关于变量的值是否是每个评估中的相同节点,或者是新节点的许可.

正如Martin所说,每次新的XSLT 3.0属性都试图控制它:如果每次调用函数时你真的想要一个新节点,你应该指定new-every-time =“yes”.

注意:

这里发生的具体优化(通过运行-explain选项可以看到)是首先内联func_2,然后将其主体提取到全局变量中.有些版本正在执行此操作,而其他版本则没有 – 它可能对较小的更改非常敏感.最好的建议是不要依赖具有这种副作用的功能.如果你解释了你的真实问题会有所帮助,那么也许我们可以找到一种对语言语义中的边缘情况不那么敏感的方法.

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


php输出xml格式字符串
J2ME Mobile 3D入门教程系列文章之一
XML轻松学习手册
XML入门的常见问题(一)
XML入门的常见问题(三)
XML轻松学习手册(2)XML概念
xml文件介绍及使用
xml编程(一)-xml语法
XML文件结构和基本语法
第2章 包装类
XML入门的常见问题(二)
Java对象的强、软、弱和虚引用
JS解析XML文件和XML字符串详解
java中枚举的详细使用介绍
了解Xml格式
XML入门的常见问题(四)
深入SQLite多线程的使用总结详解
PlayFramework完整实现一个APP(一)
XML和YAML的使用方法
XML轻松学习总节篇