Monday 12 May 2014
Monday 5 May 2014
Sunday 4 May 2014
Tail Recursion
As I work my way through my Course on Scala2, I'm learning a thing or two about functional programming. I am also learning a lot about recursion.
Some of it is also applicable to Java.
It was sufficiently new and surprising to me that I felt it deserved a blogpost. You can find all sorts of information about it elsewhere on the web as well.
What we are used to programming in Java is working iteratively, i.e. loops.
So, let us say we wish to compute the factorial3, we simply use a loop.
However, computing a Factorial has always been a very often used example to demonstrate the power of Recursion.
Unfortunately, there's a problem regarding recursion in Java.
Seeing as the method is calling itself, it keeps pushing the stack frames of the method on the stack, before calling itself. This causes a huge increase in memory, as you no doubt know. For all but trivial problems, this results in:
The above message is from the JUnit tests in my project6. It works fine, up to the last testcase, where a factorial of 10000 is requested.
The following is almost verbatim from the Scala course2:
So we need to rewrite the factorial to make use of an accumulator.
Tail recursion is often used in functional programming, and Java 8 has introduced lambdas which make functional programming a reality for Java. Therefore it is a shame that this optimization is not provided in Java (yet).
Efforts are underway to remedy this4.
I hear the syntax would look something like this:
One of the disadvantages is that this optimization makes it (a little) harder to examine stack-traces. But in my opinion, we can surmount this small obstacle easily.
Some of it is also applicable to Java.
It was sufficiently new and surprising to me that I felt it deserved a blogpost. You can find all sorts of information about it elsewhere on the web as well.
No Recursion
What we are used to programming in Java is working iteratively, i.e. loops.
So, let us say we wish to compute the factorial3, we simply use a loop.
Recursion
However, computing a Factorial has always been a very often used example to demonstrate the power of Recursion.
Unfortunately, there's a problem regarding recursion in Java.
Seeing as the method is calling itself, it keeps pushing the stack frames of the method on the stack, before calling itself. This causes a huge increase in memory, as you no doubt know. For all but trivial problems, this results in:
Testcase: factorialTest[28](FactorialTest): Caused an ERROR
java.lang.StackOverflowError
at java.math.BigDecimal.negate(BigDecimal.java:2088)
at java.math.BigDecimal.subtract(BigDecimal.java:1267)
at recursion.Recursion.factorial(Recursion.java:30)
java.lang.StackOverflowError
at java.math.BigDecimal.negate(BigDecimal.java:2088)
at java.math.BigDecimal.subtract(BigDecimal.java:1267)
at recursion.Recursion.factorial(Recursion.java:30)
The above message is from the JUnit tests in my project6. It works fine, up to the last testcase, where a factorial of 10000 is requested.
Tail Recursion
The following is almost verbatim from the Scala course2:
“In general, if the last action of a function consists of calling a function (which may be the same), such final calls are called tail calls1.”In the example of the chapter above, unfortunately, the recursion is not actually tail recursion. As you can see, after the recursion there is still the small matter of the multiplication to do. The multiplication is actually the last operation of the method.
“If a function calls itself as its last action, this is called tail recursion.”
“If a function calls itself as its last action, the function's stack frame can be reused. This is called tail recursion optimization.”
So we need to rewrite the factorial to make use of an accumulator.
“Tail recursive functions are iterative processes.”This means that a tail recursive function, if the compiler is any good, can be changed into what is effectively a loop. For more in depth information on what happens, see [5].
Notes
Scala does implement Tail Recursion Optimization. Unfortunately, Java does not. This means that in my project6 the last test case will fail in Recursion.java as well as TailRecursion.java with a StackOverflowError.Tail recursion is often used in functional programming, and Java 8 has introduced lambdas which make functional programming a reality for Java. Therefore it is a shame that this optimization is not provided in Java (yet).
Efforts are underway to remedy this4.
I hear the syntax would look something like this:
return goto factorial(accumulator.multiply(n), n.subtract(BigDecimal.ONE));
One of the disadvantages is that this optimization makes it (a little) harder to examine stack-traces. But in my opinion, we can surmount this small obstacle easily.
References
- [1] Wikipedia - Tail call
- http://en.wikipedia.org/wiki/Tail_call
- [2] Coursera - Functional Programming Principles in Scala
- https://www.coursera.org/course/progfun
- [3] Wikipedia - Factorial
- http://en.wikipedia.org/wiki/Factorial
- [4] OpenJDK Wiki - TailCalls
- https://wiki.openjdk.java.net/display/mlvm/TailCalls
- [5] Dr Dobb's - Tail Call Optimization and Java
- http://www.drdobbs.com/jvm/tail-call-optimization-and-java/240167044
- [6] GitHub - Three Java implementations of Factorial (n!) using NoRecursion, Recursion and TailRecursion
- https://github.com/maartenl/recursion
- Scala Documentation - Glossary
- http://docs.scala-lang.org/glossary/
Subscribe to:
Posts (Atom)