我正在使用 Scala 2.10.2。
我需要一个函数
def extractEither(m: M[(Key, Either[TLeft, TRight])])
: Either[TLeft, M[(Key, TRight)]]
哪里M
可以是 Seq、List、Map 或其他任何东西,并且返回类型仍然是合适的。
我正在使用以下测试:
val map = Map(1 -> Left("foo"), 2 -> Right('bar), 3 -> Right('baz))
我目前的尝试如下:
尝试#1
def extractEither[
Key, TLeft, TRight, M[_] <: TraversableOnce[_]
]
(monad: M[(Key, Either[TLeft, TRight])])
(implicit cbf: CanBuildFrom[
M[(Key, Either[TLeft, TRight])],
(Key, TRight),
M[(Key, TRight)]
]): Either[TLeft, M[(Key, TRight)]] = {
val builder = cbf(monad)
builder.sizeHint(monad.size)
(monad: GenTraversableOnce[_]).foreach { x =>
val (key, either) = x.asInstanceOf[(Key, Either[TLeft, TRight])]
either.fold(
leftVal => return Left(leftVal),
rightVal => builder += ((key, rightVal))
)
}
Right(builder.result())
}
这失败了:
scala> extractEither(map)
<console>:20: error: no type parameters for method extractEither: (monad: M[(Key, Either[TLeft,TRight])])(implicit cbf: scala.collection.generic.CanBuildFrom[M[(Key, Either[TLeft,TRight])],(Key, TRight),M[(Key, TRight)]])Either[TLeft,M[(Key, TRight)]] exist so that it can be applied to arguments (scala.collection.immutable.Map[Int,Product with Serializable with scala.util.Either[String,Symbol]])
--- because ---
argument expression's type is not compatible with formal parameter type;
found : scala.collection.immutable.Map[Int,Product with Serializable with scala.util.Either[String,Symbol]]
required: ?M
extractEither(map)
^
<console>:20: error: type mismatch;
found : scala.collection.immutable.Map[Int,Product with Serializable with scala.util.Either[String,Symbol]]
required: M[(Key, Either[TLeft,TRight])]
extractEither(map)
^
<console>:20: error: Cannot construct a collection of type M[(Key, TRight)] with elements of type (Key, TRight) based on a collection of type M[(Key, Either[TLeft,TRight])].
extractEither(map)
^
尝试#2
这个仅限于映射,可变或不可变。
def extractEither[
Key, TLeft, TRight, M <: collection.Map[Key, Either[TLeft, TRight]]
](map: M): Either[TLeft, M] = {
Right[TLeft, M](map.map { case (key, either) =>
either.fold(
leftVal => return Left(leftVal),
rightVal => key -> rightVal
)
}.asInstanceOf[M])
}
这失败了:
scala> extractEither(map)
<console>:20: error: inferred type arguments [Nothing,Nothing,Nothing,scala.collection.immutable.Map[Int,Product with Serializable with scala.util.Either[String,Symbol]]] do not conform to method extractEither's type parameter bounds [Key,TLeft,TRight,M <: scala.collection.Map[Key,Either[TLeft,TRight]]]
extractEither(map)
^
<console>:20: error: type mismatch;
found : scala.collection.immutable.Map[Int,Product with Serializable with scala.util.Either[String,Symbol]]
required: M
extractEither(map)
^
工作非通用解决方案
def extractEither[
Key, TLeft, TRight
](map: Map[Key, Either[TLeft, TRight]]): Either[TLeft, Map[Key, TRight]] = {
Right(map.map { case (key, either) =>
either.fold(
leftVal => return Left(leftVal),
rightVal => key -> rightVal
)
})
}
但这根本不是通用的:|
任何人都可以阐明如何正确编写这个吗?