[英]How do I define a function for a Monad Stack of State, Disjunction, and List?
我有一个列表monad List[Int]
。 我想根据列表的值生成效果(Disjunction和state)。 这是我的monad堆栈和运行monad堆栈的代码。 我的问题是什么应该正确定义checkNum以便我可以生成正确的效果?
我的预期产量应该是
List(("", \\/-(1), ("Error: 2", -\\/(Throwable())), ("",\\/-(3)), ("Error: 4", -\\/(Throwable())))
import scalaz._
import Scalaz._
type EitherTH[F[_], A] = EitherT[F, Throwable,A]
type StateTH[F[_], A] = StateT[F, String, A]
type StateTList[A] = StateTH[List, A]
type EitherTStateTList[A] = EitherTH[StateTList, A]
val lst = List(1,2,3,4)
def checkNum(x:Int)(implicit ms:MonadState[EitherTStateTList, Int]) = if ((x%2)==0) {
put(s"Error: $x")
-\/(new Throwable())
} else {
put("")
\/-(x)
}
val prg = for {
x <- lst.liftM[StateTH].liftM[EitherTH]
// y <- checkNum(x).liftM[EitherTH]
} yield y
prg.run("")
在我看来, checkNum
应该返回一个State[String, \\/[Throwable,Int]]
:
def checkNum0(x: Int): State[String, \/[Throwable,Int]] = if ((x%2)==0) {
constantState(-\/(new Throwable()), s"Error: $x")
} else {
constantState(\/-(x), "")
}
def checkNum1(x: Int): StateTList[\/[Throwable,Int]] = checkNum0(x).lift[List]
然后你可以把你的理解写成:
val prg = for {
x <- lst.liftM[StateTH].liftM[EitherTH]
y <- EitherT(checkNum1(x))
} yield y
输出:
List(
("", \/-(1)),
("Error: 2", -\/(java.lang.Throwable)),
("", \/-(3)),
("Error: 4", -\/(java.lang.Throwable))
)
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.