[Groovy] Annotation Type TailRecursive
- groovy.transform.TailRecursive
- All Implemented Interfaces and Traits:
- Annotation
@Retention(value: RetentionPolicy.SOURCE) @Target(value: [ElementType.METHOD]) @GroovyASTTransformationClass(value: [org.codehaus.groovy.transform.tailrec.TailRecursiveASTTransformation]) @interface TailRecursive
Method annotation used to transform methods with tail recursive calls into iterative methods automagically since the JVM cannot do this itself. This works for both static and non-static methods.
It allows you to write a method like this:import groovy.transform.TailRecursive class Target {@TailRecursive
long sumUp(long number, long sum = 0) { if (number == 0) return sum; sumUp(number - 1, sum + number) } } def target = new Target() assert target.sumUp(100) == 5050 assert target.sumUp(1000000) == 500000500000 //will blow the stack on most machines when used without@TailRecursive
@TailRecursive
is supposed to work in combination with @CompileStatic
Known shortcomings: - Only non-void methods are currently being handled. Void methods will fail compilation.
- Only direct recursion (calling the exact same method again) is supported.
- Mixing of tail calls and non-tail calls is not possible. The compiler will complain if some recursive calls cannot be handled.
- Checking if a recursive call is really tail-recursive is not very strict. You might run into cases where non-tail calls will be considered tail calls.
- In the presence of method overloading and method overriding you might run into situations where a call is considered recursive although it really is not.
- Catching
Throwable
around a recursive might lead to problems - Non trivial continuation passing style examples do not work.
- Probably many unrecognized edge cases.
More examples:
import groovy.transform.TailRecursive @TailRecursive long sizeOfList(list, counter = 0) { if (list.size() == 0) { counter } else { sizeOfList(list.tail(), counter + 1) } } // Without @TailRecursive a StackOverFlowError // is thrown. assert sizeOfList(1..10000) == 10000
- Since:
- 2.3
Inherited Methods Summary
Methods inherited from class | Name |
---|---|
class Object | wait, wait, wait, equals, toString, hashCode, getClass, notify, notifyAll |
© 2003-2020 The Apache Software Foundation
Licensed under the Apache license.
https://docs.groovy-lang.org/3.0.7/html/gapi/groovy/transform/TailRecursive.html