我使用arrow-kt库,并且尝试在同一库中使用Either
和IO
进行理解。
说我有下一段代码:
IO.monad().binding {
val ans: Either<Error, Data> = someFunctionThatReturnsEitherWrappedInIO().bind()
}
现在,我想在ans
上使用绑定:
val data: Data = ans.bind() // My intent
在第一段代码的范围内是否有可能做到这一点?
当前,我正在尝试将一个Either
绑定嵌套在IO绑定的范围内,但是我不确定这是一个好习惯:
IO.monad().binding {
val ans: Either<Error, Data> = someFunctionThatReturnsEitherWrappedInIO().bind()
val ansB: Either<Error, OtherData> = someOtherFunctionThatReturnsEitherWrappedInIO().bind()
val newData: Either<Any, NewData> = Either.monad<Any>().binding {
val data: Data = ans.bind()
val otherData: OtherData = ansB.bind()
NewData(data.a, otherData.lala)
}.fix()
}
答案 0 :(得分:5)
首先我要提到Monads don't compose,这就是为什么您需要Monad transformer的原因,在您的情况下EitherT
是可以帮助您的人。
object Error
fun one() = IO { Right(1) }
fun two() = IO { Right("2") }
fun toInt(str: String) = IO { Try { str.toInt() }.toEither { Error } }
val result: IO<Either<Error, Int>> =
EitherT.monad<ForIO, Error>(IO.monad()).binding {
val oneInt = EitherT(one()).bind()
val twoString = EitherT(two()).bind()
val twoInt = EitherT(toInt(twoString)).bind()
oneInt + twoInt
}.value().fix()
println(result.unsafeRunSync()) // Just for demonstration, don't do this ever
右(b = 3)