私は、talkで次の例を試して、java8の末尾再帰を理解していました。ここでTailCalljava 8を使用したテール再帰の設計
public class TailCalls {
public static <T> TailCall<T> call(final TailCall<T> nextcall) {
return nextcall;
}
public static <T> TailCall<T> done(final T value) {
return new TailCall<T>() {
@Override
public boolean isComplete() {
return true;
}
@Override
public T result() {
return value;
}
@Override
public TailCall<T> apply() {
throw new Error("not implemented.");
}
};
}
}
を使用する
@FunctionalInterface
public interface TailCall<T> {
TailCall<T> apply();
default boolean isComplete() {
return false;
}
default T result() {
throw new Error("not implemented");
}
default T get() {
return Stream.iterate(this, TailCall::apply).filter(TailCall::isComplete)
.findFirst().get().result();
}
}
ユーティリティクラスは、末尾再帰の使用は、次のとおりです。
public class Main {
public static TailCall<Integer> factorial(int fact, int n) {
if (n == 1) {
return TailCalls.done(fact);
} else {
return TailCalls.call(factorial(fact * n, n-1));
}
}
public static void main(String[] args) {
System.out.println(factorial(1, 5).get());
}
}
それは正しく働いたが、我々は必要としないような気がしますTailCall::get
を使用して結果を計算します。私の理解あたりとして、我々は直接使用して結果を計算することができます
System.out.println(factorial(1, 5).result());
の代わり:
System.out.println(factorial(1, 5).get());
を私はTailCall::get
の要旨をしないのです場合は私に知らせてください。
fwiwテクニックは[トランポリング(trampolining)](https://en.wikipedia.org/wiki/Trampoline_(computing))と呼ばれ、このテクニックを使用して反復を行うことができますスタックセーフであるため、末尾再帰をサポートしない言語を使用するプログラムでは、実際のテールコール除去を実装できません。 – naomik