【问题标题】:CPS/Continuations StackOverflowError on (tail-)recursive functions(尾)递归函数上的 CPS/Continuations StackOverflowError
【发布时间】:2011-02-18 02:47:19
【问题描述】:

有什么方法可以让 CPS 中的尾递归函数不抛出 StackOverflow?

import scala.util.continuations._
object CPSStackOverflow {
  def main(args: Array[String]) = {
    reset {
      def recurse(i: Int): Unit @suspendable = {
        println(i)
        shift { k: (Unit => Unit) =>
          k( () ) // just continue
        }
        recurse(i + 1)
      }
      recurse(1)
    }
  }
}

导致以下 StackOverflowError:

1298
1299
1300
Exception in thread "main" java.lang.StackOverflowError
    at java.nio.CharBuffer.wrap(CharBuffer.java:350)
    at sun.nio.cs.StreamEncoder.implWrite(StreamEncoder.java:246)
    at sun.nio.cs.StreamEncoder.write(StreamEncoder.java:106)
    at java.io.OutputStreamWriter.write(OutputStreamWriter.java:190)
    at java.io.BufferedWriter.flushBuffer(BufferedWriter.java:111)
    at java.io.PrintStream.write(PrintStream.java:476)
    at java.io.PrintStream.print(PrintStream.java:619)
    at java.io.PrintStream.println(PrintStream.java:773)
    at scala.Console$.println(Console.scala:198)
    at scala.Predef$.println(Predef.scala:182)
    at test.CPSStackOverflow$$anonfun$main$1.recurse$1(CPSStackOverflow.scala:9)
    at test.CPSStackOverflow$$anonfun$main$1$$anonfun$recurse$1$2.apply(CPSStackOverflow.scala:13)
    at test.CPSStackOverflow$$anonfun$main$1$$anonfun$recurse$1$2.apply(CPSStackOverflow.scala:10)
    at scala.util.continuations.ControlContext$$anonfun$flatMap$2$$anonfun$apply$2.apply(ControlContext.scala:71)
    at test.CPSStackOverflow$$anonfun$main$1$$anonfun$recurse$1$1.apply(CPSStackOverflow.scala:11)
    at test.CPSStackOverflow$$anonfun$main$1$$anonfun$recurse$1$1.apply(CPSStackOverflow.scala:10)
    at scala.util.continuations.package$$anonfun$shiftR$1.apply(package.scala:58)
    at scala.util.continuations.package$$anonfun$shiftR$1.apply(package.scala:58)
    at scala.util.continuations.ControlContext$$anonfun$flatMap$2.apply(ControlContext.scala:68)
    at scala.util.continuations.ControlContext$$anonfun$flatMap$2.apply(ControlContext.scala:67)
    at scala.util.continuations.ControlContext$$anonfun$flatMap$2$$anonfun$apply$2.apply(ControlContext.scala:73)
    at test.CPSStackOverflow$$anonfun$main$1$$anonfun$recurse$1$1.apply(CPSStackOverflow.scala:11)
    at test.CPSStackOverflow$$anonfun$main$1$$anonfun$recurse$1$1.apply(CPSStackOverflow.scala:10)
    at scala.util.continuations.package$$anonfun$shiftR$1.apply(package.scala:58)
    at scala.util.continuations.package$$anonfun$shiftR$1.apply(package.scala:58)
etc...

有什么办法可以规避这个错误?蹦床?通过抛出异常来展开堆栈? 谢谢!

【问题讨论】:

  • 至少 while 循环虽然被 CPS-compiler-plugin 重写,但不会杀死堆栈...

标签: scala scala-2.8 continuations


【解决方案1】:

您可以使用-Xss2M 运行 Java,但是该错误可能仅在一千次迭代之后才会发生。只要你的方法不是尾递归的,你就无法解决这个问题。

【讨论】:

    【解决方案2】:

    您在延续中调用另一个函数。 Scala 不支持跨方法尾递归,因为 JVM 不支持。

    【讨论】:

    • recurse 上放置一个@tailrec 注释,编译器会在它无法满足您的尾递归请求时告诉您。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2019-09-14
    • 2016-05-06
    • 2014-02-03
    • 1970-01-01
    • 2012-12-28
    • 2016-12-20
    • 1970-01-01
    相关资源
    最近更新 更多