下面是如何使用“Manifet”在运行时创建类型为T的新实例:
上面的
只要我在
但是一旦我定义了一个继承自
我遇到了以下错误:
trait MyTrait
class MyClass1(val name: String) extends MyTrait
class MyClass2(val name: String) extends MyTrait
class Test[T <: MyTrait] {
def createInstance[T](name: String)(implicit m: Manifest[T]): T = {
m.runtimeClass.getConstructors()(0)
.newInstance(name).asInstanceOf[T]
}
def doSomething() {
val myClass = createInstance("joe")
...
}
}
...
val test = new Test[MyClass1]
test.doSomething
上面的
createInstance
方法创建了一个实现MyTrait
接口的类的新实例,并使用给定的字符串调用构造函数。如何使用TypeTag
实现相同的功能?
使用scala.reflect.runtime._重新实现
下面是按照som-snytt的建议重新实现的Test类:class Test[T <: MyTrait] {
import scala.reflect.runtime._
import scala.reflect.runtime.universe._
def createInstance[T: TypeTag](name: String): T = {
val tt = typeTag[T]
currentMirror.reflectClass(tt.tpe.typeSymbol.asClass).reflectConstructor(
tt.tpe.members.filter(m =>
m.isMethod && m.asMethod.isConstructor
).iterator.next.asMethod
)(name).asInstanceOf[T]
}
}
只要我在
Test
上调用 createInstance
就可以正常运作:val test = new Test[MyClass1]
val myClass = test.createInstance("hello") // this works
但是一旦我定义了一个继承自
Test
的新类...class DerivedTest[T <: MyTrait] extends Test[T]
我创建了一个新类,然后使用createInstance
方法调用该类...
val test = new DerivedText[MyClass1]
val myClass = test.createInstance("hello") // this crashes
我遇到了以下错误:
...
java.util.NoSuchElementException: next on empty iterator
at scala.collection.Iterator$$anon$2.next(Iterator.scala:39)
at scala.collection.Iterator$$anon$2.next(Iterator.scala:37)
at scala.collection.LinearSeqLike$$anon$1.next(LinearSeqLike.scala:62)
at DerivedTest$class.createInstance(<console>:27)
at $anon$1.createInstance(<console>:26)
at .<init>(<console>:28)
at .<clinit>(<console>)
at .<init>(<console>:7)
at .<clinit>(<console>)
at $print(<console>)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:606)
...
我有什么遗漏吗?
createInstance
方法中的T
遮蔽了Test
类中的T
。这可能不是您想要的。 - ghik