2

我想创建一个g将函数f作为参数的函数,其中f有一个类型参数。我无法获得g该编译的签名。一种尝试是这样的:

scala> def mock1[A](): A = null.asInstanceOf[A] // STUB
mock1: [A]()A

scala> def mock2[A](): A = null.asInstanceOf[A] // STUB
mock2: [A]()A

scala> def g0(f: [A]() => A): Int = f[Int]()
<console>:1: error: identifier expected but '[' found.
       def g0(f: [A]() => A): Int = f[Int]()
                 ^

如果我将接受类型参数的函数包装在特征中,我可以让它工作,如下所示:

scala> trait FWrapper { def f[A](): A }
defined trait FWrapper

scala> class mock1wrapper extends FWrapper { def f[A]() = mock1[A]() }
defined class mock1wrapper

scala> class mock2wrapper extends FWrapper { def f[A]() = mock2[A]() }
defined class mock2wrapper

scala> def g(wrapper: FWrapper): Int = wrapper.f[Int]()
g: (wrapper: FWrapper)Int

scala> g(new mock1wrapper)
res8: Int = 0

有没有一种方法可以在不引入包装类的情况下实现这一点?

4

2 回答 2

6

Scala 确实(当前)不支持多态函数值。你有两个选择:

  1. 坚持你的包装特性(可能最容易理解)
  2. 使用Shapeless 中的多态函数值(花哨,但可能有点复杂)
于 2013-06-03T20:19:29.337 回答
0

这个怎么样:

  def mock[A](): A = null.asInstanceOf[A]

  def g[A](f:() => A): A = f()

  g(mock[Int])
于 2013-06-03T19:33:08.800 回答