19

看来我不明白一些重要的事情,也许是关于擦除(该死)。

我有一个方法,我想创建一个大小数组,其中n填充了以下值gen

def testArray[T](n: Int, gen: =>T) {
  val arr = Array.fill(n)(gen)
  ...
}

并使用它,例如:

testArray(10, util.Random.nextInt(10))

但我得到错误:

scala: could not find implicit value for evidence parameter of type scala.reflect.ClassManifest[T]
val arr = Array.fill(n)(gen)
                       ^

请解释我做错了什么,为什么会出现这个错误,以及它使什么样的代码变得不可能?

4

2 回答 2

14

那是因为在编译时不知道testArray具体类型。T您的签名必须看起来像def testArray[T : ClassManifest](n: Int, gen: =>T),这将为您的方法添加一个类型的隐式参数,该参数ClassManifest[T]会自动传递给调用,testArray然后进一步传递给Array.fill调用。这称为context bound.

于 2013-02-10T10:14:59.473 回答
5

Array.fill方法具有以下签名:

def fill[T](n: Int)(elem: => T)(implicit arg0: ClassManifest[T]): Array[T]

为了获得一个实例,ClassManifest[T]您需要知道具体类型。AClassManifest可以这样获得:

implicitly[ClassManifest[String]]

AClassManifest对于每种具体类型都是隐式可用的。

对于任何implicit错误,您可以使用 type 参数将所需的隐式添加到方法中:

def wrap[T](n:Int)(elem: => T)(implicit c:ClassManifest[T], o:Ordering[T])

如果您自己没有介绍ClassManifestor Ordering,那么库的作者(很可能)为您提供了合理的默认值。

如果您要调用该wrap方法:

wrap(2)(3)

它是这样展开的:

wrap[Int](2)(3)(implicitly[ClassManifest[Int]], implicitly[Ordering[Int]])

如果你Person在这里引入了一个自定义类,你会因为找不到Ordering[Person]. 图书馆的作家不可能知道如何订购Person。你可以这样解决:

class Person

implicit val o = new Ordering[Person] { // implement required methods }

wrap(2)(new Person)

Scala 编译器在不同的范围内寻找隐式,Ordering通常不会像这样指定。我建议您在互联网上查找隐式分辨率以了解更多信息。

于 2013-02-10T12:19:10.997 回答