对Kotlin和arrow-kt库的理解

时间:2018-12-12 15:58:59

标签: kotlin io functional-programming either arrow-kt

我使用arrow-kt库,并且尝试在同一库中使用EitherIO进行理解。

说我有下一段代码:

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()
}

1 个答案:

答案 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)