Considerare questo codice (preso da here e modificato per utilizzare byte anziché righe di caratteri).Come utilizzare IO con Scalaz7 Iteratees senza sovraccaricare lo stack?
import java.io.{ File, InputStream, BufferedInputStream, FileInputStream }
import scalaz._, Scalaz._, effect._, iteratee.{ Iteratee => I, _ }
import std.list._
object IterateeIOExample {
type ErrorOr[+A] = EitherT[IO, Throwable, A]
def openStream(f: File) = IO(new BufferedInputStream(new FileInputStream(f)))
def readByte(s: InputStream) = IO(Some(s.read()).filter(_ != -1))
def closeStream(s: InputStream) = IO(s.close())
def tryIO[A, B](action: IO[B]) = I.iterateeT[A, ErrorOr, B] {
EitherT(action.catchLeft).map(r => I.sdone(r, I.emptyInput))
}
def enumBuffered(r: => BufferedInputStream) = new EnumeratorT[Int, ErrorOr] {
lazy val reader = r
def apply[A] = (s: StepT[Int, ErrorOr, A]) => s.mapCont(k =>
tryIO(readByte(reader)) flatMap {
case None => s.pointI
case Some(byte) => k(I.elInput(byte)) >>== apply[A]
})
}
def enumFile(f: File) = new EnumeratorT[Int, ErrorOr] {
def apply[A] = (s: StepT[Int, ErrorOr, A]) =>
tryIO(openStream(f)).flatMap(stream => I.iterateeT[Int, ErrorOr, A](
EitherT(
enumBuffered(stream).apply(s).value.run.ensuring(closeStream(stream)))))
}
def main(args: Array[String]) {
val action = (
I.consume[Int, ErrorOr, List] &=
enumFile(new File(args(0)))).run.run
println(action.unsafePerformIO())
}
}
L'esecuzione di questo codice su un file di dimensioni decenti (8kb) produce una StackOverflowException. Alcune ricerche hanno rivelato che l'eccezione poteva essere evitata usando la monade Trampoline invece di IO, ma questa non sembra una grande soluzione - sacrificare la purezza funzionale per completare il programma. Il modo ovvio per risolvere questo problema è usare IO o Trampoline come un trasformatore di Monade per avvolgere l'altro, ma non riesco a trovare un'implementazione della versione del trasformatore di nessuno di essi e non sono abbastanza un guru della programmazione funzionale per so come scrivere il mio (imparare di più su FP è uno degli scopi di questo progetto, ma sospetto che la creazione di nuovi trasformatori monad sia un po 'al di sopra del mio livello al momento). Suppongo che potrei semplicemente avvolgere una grande azione IO attorno alla creazione, all'esecuzione e alla restituzione del risultato dei miei iterate, ma mi sembra più una soluzione alternativa che una soluzione.
Presumibilmente alcune monade non possono essere convertite in trasformatori monad, quindi mi piacerebbe sapere se è possibile lavorare con file di grandi dimensioni senza perdere IO o sovraccaricare lo stack, e se sì, come?
Domanda bonus: Non riesco a pensare a un modo per iteratee per segnalare che è verificato un errore durante l'elaborazione fatta eccezione per averlo tornare entrambi i casi, che lo rende meno facile da comporre loro. Il codice sopra mostra come usare EitherT per gestire gli errori nell'enumeratore, ma come funziona per gli iterate?
Questo potrebbe essere utile a voi: http://termsandtruthconditions.herokuapp.com/blog/2013/03/16/free-monad/ – Impredicative
È una buona spiegazione del motivo per cui ho bisogno di usare il trampolino per evitare di traboccare lo stack, ma non copre come utilizzare sia l'IO che il trampolino. – Redattack34
IO è già trampolino. – Apocalisp