2012-12-16 26 views
3

我想在XSLT 2.0中编写一个尾递归函数,该函数遍历一个多值变量的日期并返回最早的一个。出于某种原因,我的功能不受SaxonHE9.4认定为尾递归,我得到以下错误,当输入文件已经超过150-200项左右:在tail_rec_test线73XSLT 2.0中的尾递归函数不起作用

错误。 xsl:调用太多嵌套函数 。可能是由于无限递归。在内置模板规则

这里是我的XML输入:

<?xml version="1.0"?> 
<Events> 
    <Event> 
    <Date>2004-01-01</Date> 
    </Event> 
    <Event> 
    <Date>2003-01-01</Date> 
    </Event> 
    <Event> 
    <Date>2002-01-01</Date> 
    </Event> 
    <Event> 
    <Date>2001-01-01</Date> 
    </Event> 
    <Event> 
    <Date>2005-01-01</Date> 
    </Event> 
    <Event> 
    <Date>2006-01-01</Date> 
    </Event> 
    <Event> 
    <Date>2007-01-01</Date> 
    </Event> 
    <Event> 
    <Date>2008-01-01</Date> 
    </Event> 
</Events> 

这是我的XSL文件看起来像:

<xsl:stylesheet version="2.0" 
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:fo="http://www.w3.org/1999/XSL/Format" 
    xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:fn="http://www.w3.org/2005/xpath-functions" 
    xmlns:own="http://ownfunctions"> 
    <xsl:output method="xml" indent="yes"/> 
    <xsl:strip-space elements="*"/> 


    <xsl:function name="own:findEarliestDate"> 
     <xsl:param name="dates" as="xs:date*"/> 

     <xsl:variable name="size"><xsl:value-of select="count($dates)" /></xsl:variable> 
     <xsl:choose> 
      <xsl:when test="$size &gt; 0"> 
       <xsl:value-of select="own:findEarliestDate-helper($dates, $size, xs:date('2050-01-01'))" /> 
      </xsl:when> 
      <xsl:otherwise> 
       <xsl:value-of select="''"/> 
      </xsl:otherwise> 
     </xsl:choose> 
    </xsl:function> 


    <xsl:function name="own:findEarliestDate-helper" as="xs:date"> 
     <xsl:param name="items" as="xs:date*"/> 
     <xsl:param name="i" as="xs:integer"/> 
     <xsl:param name="curMin" as="xs:date"/> 

     <xsl:choose> 
      <xsl:when 
       test="$i = 0"> 
       <xsl:value-of select="xs:date($curMin)"/>   
      </xsl:when> 
      <xsl:otherwise> 
       <xsl:variable name="item" as="xs:date"> 
        <xsl:value-of select="xs:date($items[$i])"/> 
       </xsl:variable> 

       <xsl:variable name="next" as="xs:date"> 
        <xsl:choose> 
         <xsl:when test="$item &lt; $curMin"> 
          <xsl:value-of select="$item"/> 
         </xsl:when> 
         <xsl:otherwise> 
          <xsl:value-of select="$curMin"/> 
         </xsl:otherwise> 
        </xsl:choose> 
       </xsl:variable> 

       <xsl:value-of select="own:findEarliestDate-helper($items, $i - 1, $next)"/> 
      </xsl:otherwise> 
     </xsl:choose> 
    </xsl:function> 


    <xsl:template match="Events"> 
     <xsl:variable name="items" as="xs:date*"> 
      <xsl:for-each select="Event"> 
       <xsl:value-of select="xs:date(Date)"/> 
      </xsl:for-each> 
     </xsl:variable> 

     <Test> 
      <EarliestDate> 
       <xsl:value-of select="own:findEarliestDate($items)"/> 
      </EarliestDate> 
     </Test> 
    </xsl:template> 

</xsl:stylesheet> 

我怎样才能变换它变成了一个正确的尾递归函数? 我已经测试了这个例子,但我不能将其应用到自己的代码: http://www.nesterovsky-bros.com/weblog/2008/02/20/EfficientXslt20RecursionInSaxon9.aspx

回答

5

我不能瑞普这个

使用撒克逊9.4.06EE(试用版),结果是:

<Test xmlns:fo="http://www.w3.org/1999/XSL/Format" 
     xmlns:xs="http://www.w3.org/2001/XMLSchema" 
     xmlns:fn="http://www.w3.org/2005/xpath-functions" 
     xmlns:own="http://ownfunctions"> 
    <EarliestDate>2001-01-01</EarliestDate> 
</Test> 

Saxon-EE 9.4.0.6J from Saxonica 
Java version 1.6.0_31 
Using license serial number XXXXXXXX 
Generating byte code... 
Stylesheet compilation time: 2168 milliseconds 
Processing file:/C:/Program%20Files/Java/jre6/bin/marrowtr.xml 
Using parser com.sun.org.apache.xerces.internal.jaxp.SAXParserImpl$JAXPSAXParser 
Building tree for file:/C:/Program%20Files/Java/jre6/bin/marrowtr.xml using class net.sf.saxon.tree.tiny.TinyBuilder 
Tree built in 10 milliseconds 
Tree size: 27 nodes, 80 characters, 0 attributes 
Execution time: 122ms 
Memory used: 52169472 
NamePool contents: 8 entries in 8 chains. 6 URIs 

BTW,一个可以只使用它来产生相同的结果

<xsl:value-of select="min(/*/*/Date/xs:date(.))"/> 

更新

问题是在这条线:

  <xsl:value-of select="own:findEarliestDate-helper($items, $i - 1, $next)"/> 

因为函数的返回类型是xs:date,上面的行是不是在功能的执行顺序的最后一行。它产生一个字符串(更准确地说,是一个文本节点),XSLT处理器需要获取该字符串并将其转换为xs:date - 这意味着该函数占用的内存不会被丢弃,堆栈继续增长,直到溢出。

的解决方案是简单

替换上面:

  <xsl:sequence select="own:findEarliestDate-helper($items, $i - 1, $next)"/> 

这产生xs:date和XSLT处理器现在可识别的功能的尾递归。

我测试了1000个事件(原始代码崩溃)和结果产生正常(和更快)的正确代码。

Saxon-EE 9.4.0.6J from Saxonica 
Java version 1.6.0_31 
Using license serial number XXXXXXXXXX 
Generating byte code... 
Stylesheet compilation time: 2002 milliseconds 
Processing file:/C:/Program%20Files/Java/jre6/bin/marrowtr.xml 
Using parser com.sun.org.apache.xerces.internal.jaxp.SAXParserImpl$JAXPSAXParser 
Building tree for file:/C:/Program%20Files/Java/jre6/bin/marrowtr.xml using class net.sf.saxon.tree.tiny.TinyBuilder 
Tree built in 124 milliseconds 
Tree size: 3032 nodes, 9800 characters, 0 attributes 
Execution time: 364ms 
Memory used: 55089048 
NamePool contents: 8 entries in 8 chains. 6 URIs 
+0

感谢您的回复速度快(+1)!我测试了短版本,它绝对有效(5000个事件很有效)。虽然很高兴知道为什么我的函数不像尾递归函数那样工作。它在150-200个事件中搜索,但这可能取决于VM堆大小。你可以请尝试很多事件吗?我可以更新输入,但它会很大。只要说你喜欢什么。再次感谢。 – Ivo

+0

@Ivo,我发现你的问题,并用问题,解释和简单的解决方案更新了我的答案。 –

+0

太棒了!它现在完美运行,谢谢! – Ivo