aşağıdaki blog article F # foldBack
idame geçen stilini kullanarak kuyruk özyinelemeli yapılabilir gösterir.foldRight kuyruğunu tekrarlamak için devamlılık kullanmak mümkün mü?
Scala bu anlamına gelir:
def foldBack[T,U](l: List[T], acc: U)(f: (T, U) => U): U = {
l match {
case x :: xs => f(x, foldBack(xs, acc)(f))
case Nil => acc
}
}
yaparak kuyruk özyinelemeli yapılabilir:
def foldCont[T,U](list: List[T], acc: U)(f: (T, U) => U): U = {
@annotation.tailrec
def loop(l: List[T], k: (U) => U): U = {
l match {
case x :: xs => loop(xs, (racc => k(f(x, racc))))
case Nil => k(acc)
}
}
loop(list, u => u)
}
Ne yazık ki, hala uzun listeler için bir yığın taşması olsun. döngü kuyruk özyinelemeli ve optimize edildi ama sanırım yığın birikimi sadece devam çağrılarına taşındı.
Neden bu F # ile bir sorun değildir? Ve Scala ile bunun etrafında çalışmak için herhangi bir yolu var mı?
Düzenleme: burada bazı kod yığınının derinliğini göstermektedir:
def showDepth(s: Any) {
println(s.toString + ": " + (new Exception).getStackTrace.size)
}
def foldCont[T,U](list: List[T], acc: U)(f: (T, U) => U): U = {
@annotation.tailrec
def loop(l: List[T], k: (U) => U): U = {
showDepth("loop")
l match {
case x :: xs => loop(xs, (racc => { showDepth("k"); k(f(x, racc)) }))
case Nil => k(acc)
}
}
loop(list, u => u)
}
foldCont(List.fill(10)(1), 0)(_ + _)
Bu baskılar:
loop: 50
loop: 50
loop: 50
loop: 50
loop: 50
loop: 50
loop: 50
loop: 50
loop: 50
loop: 50
loop: 50
k: 51
k: 52
k: 53
k: 54
k: 55
k: 56
k: 57
k: 58
k: 59
k: 60
res2: Int = 10
Mantıklı değil. Basit bir test vaktin var mı? –
@ DanielC.Sobral, eklediğim kodu görüyorum ve yazdırıyor. – huynhjl