这里有两个问题。首先是您的函数的推断类型过于具体。Option
是一个单子,但Some
不是。在像 Haskell 这样的语言中,等价物Some
甚至不是类型——它只是一个构造函数——但由于代数数据类型在 Scala 中的编码方式,你必须注意这个问题。有两个简单的修复——要么明确地提供更通用的类型:
scala> val f: Int => Option[Int] = i => Some(i + 1)
f: Int => Option[Int] = <function1>
scala> val g: Int => Option[String] = i => Some(i.toString)
g: Int => Option[String] = <function1>
或者使用Scalaz的handy some
,它返回一个适当类型的Some
:
scala> val f = (i: Int) => some(i + 1)
f: Int => Option[Int] = <function1>
scala> val g = (i: Int) => some(i.toString)
g: Int => Option[String] = <function1>
第二个问题是>=>
Scalaz 中没有为普通的旧 monadic 函数提供它——您需要使用Kleisli
包装器:
scala> val h = Kleisli(f) >=> Kleisli(g)
h: scalaz.Kleisli[Option,Int,String] = Kleisli(<function1>)
这正是你想要的——只是h.run
用来展开。