如果我有多个操作返回Validation[E, _]
具有固定错误类型的内容,我可以在for-comprehension中使用它们。例如:
val things: Validation[E, (Int, Double)] = for {
i <- getValidationOfInt
d <- getValidationOfDouble
} yield (i, d)
如果错误类型不同,怎么办?假设我从HTTP读取并希望将字符串响应转换为Int
。
import scalaz._; import Scalaz._
object ValidationMixing {
class HttpError
def getFromHttp: Validation[HttpError, String] = ???
def parseInt(json: String): Validation[Throwable, Int] =
Validation.fromTryCatchNonFatal(Integer.parseInt(json))
val intParsedFromHttp: Validation[Any, Int] = for {
s <- getFromHttp
i <- parseInt(s)
} yield i
}
这是编译,但只是因为验证的错误类型是Any
,是Throwable
和HttpError
的超类型。这不是非常有用。
我可以想出各种方式来表示比Any
更有用的组合错误类型(例如Validation[Error1 \/ Error2, Result]
来存储,Validation[String, Result]
转换为错误消息等)但它们都有缺点。
有没有惯用的方法呢?
答案 0 :(得分:1)
由于没有人有更好的主意,我会留下我的答案以供将来参考。
如评论中所述,最好的方法是创建错误层次结构:
trait GenericError { /* some commond fields */}
case class MyNumericError(/* fields */)
然后在验证时使用leftMap
来生成相应的错误:
Validation.fromTryCatchNonFatal(...).leftMap(t => MyNumericError(...))
这种方法有两个优点
Validation[GenericError, T]
,因此在验证的左侧部分没有不同的类型