使用仿函数 (Scalaz7) 提升采用隐式参数的函数
Lifting a function which takes implicit parameter using functor (Scalaz7)
刚开始学习 Scalaz。这是我的代码
trait Monoid[A] {
def mappend(a1: A, a2: A): A
def mzero: A
}
object Monoid {
implicit val IntMonoid: Monoid[Int] = new Monoid[Int] {
def mappend(a1: Int, a2: Int): Int = a1 + a2
def mzero: Int = 0
}
implicit val StringMonoid: Monoid[String] = new Monoid[String] {
def mappend(a1: String, a2: String): String = a1 + a2
def mzero: String = ""
}
}
trait MonoidOp[A] {
val F: Monoid[A]
val value: A
def |+|(a2: A): A = F.mappend(value, a2)
}
object MonoidOp{
implicit def toMonoidOp[A: Monoid](a: A): MonoidOp[A] = new MonoidOp[A]{
val F = implicitly[Monoid[A]]
val value = a
}
}
我已经定义了一个函数(只是为了它)
def addXY[A: Monoid](x: A, y: A): A = x |+| y
我想解除它以便可以使用 Option、List 等容器来使用它。但是当我这样做时
def addXYOptioned = Functor[Option].lift(addXY)
它说 error: could not find implicit value for evidence parameter of type scalaz.Monoid[A]
def addOptioned = Functor[Option].lift(addXY)
如何解除这些功能?
您的方法 addXY
需要一个 Monoid[A]
但在 addXYOptioned
中使用时范围内没有 Monoid[A]
,因此您还需要添加 Monoid
约束到 addXYOptioned
.
下一个问题是Functor.lift
only lifts a function A => B
, but we can use Apply.lift2
解除一个函数(A, B) => C
。
使用 Scalaz 本身的 Monoid
:
import scalaz._, Scalaz._
def addXY[A: Monoid](x: A, y: A): A = x |+| y
def addXYOptioned[A: Monoid] = Apply[Option].lift2(addXY[A] _)
我们可以概括 addXYOptioned
以将 addXY
提升到具有 Apply
实例的任何类型构造函数中:
def addXYApply[F[_]: Apply, A: Monoid] = Apply[F].lift2(addXY[A] _)
addXYApply[List, Int].apply(List(1,2), List(3,4))
// List[Int] = List(4, 5, 5, 6)
addXYApply[Option, Int].apply(1.some, 2.some)
// Option[Int] = Some(3)
刚开始学习 Scalaz。这是我的代码
trait Monoid[A] {
def mappend(a1: A, a2: A): A
def mzero: A
}
object Monoid {
implicit val IntMonoid: Monoid[Int] = new Monoid[Int] {
def mappend(a1: Int, a2: Int): Int = a1 + a2
def mzero: Int = 0
}
implicit val StringMonoid: Monoid[String] = new Monoid[String] {
def mappend(a1: String, a2: String): String = a1 + a2
def mzero: String = ""
}
}
trait MonoidOp[A] {
val F: Monoid[A]
val value: A
def |+|(a2: A): A = F.mappend(value, a2)
}
object MonoidOp{
implicit def toMonoidOp[A: Monoid](a: A): MonoidOp[A] = new MonoidOp[A]{
val F = implicitly[Monoid[A]]
val value = a
}
}
我已经定义了一个函数(只是为了它)
def addXY[A: Monoid](x: A, y: A): A = x |+| y
我想解除它以便可以使用 Option、List 等容器来使用它。但是当我这样做时
def addXYOptioned = Functor[Option].lift(addXY)
它说 error: could not find implicit value for evidence parameter of type scalaz.Monoid[A]
def addOptioned = Functor[Option].lift(addXY)
如何解除这些功能?
您的方法 addXY
需要一个 Monoid[A]
但在 addXYOptioned
中使用时范围内没有 Monoid[A]
,因此您还需要添加 Monoid
约束到 addXYOptioned
.
下一个问题是Functor.lift
only lifts a function A => B
, but we can use Apply.lift2
解除一个函数(A, B) => C
。
使用 Scalaz 本身的 Monoid
:
import scalaz._, Scalaz._
def addXY[A: Monoid](x: A, y: A): A = x |+| y
def addXYOptioned[A: Monoid] = Apply[Option].lift2(addXY[A] _)
我们可以概括 addXYOptioned
以将 addXY
提升到具有 Apply
实例的任何类型构造函数中:
def addXYApply[F[_]: Apply, A: Monoid] = Apply[F].lift2(addXY[A] _)
addXYApply[List, Int].apply(List(1,2), List(3,4))
// List[Int] = List(4, 5, 5, 6)
addXYApply[Option, Int].apply(1.some, 2.some)
// Option[Int] = Some(3)