gpt4 book ai didi

scala - 如何在运行时使用 TypeTags 创建类型 T 的实例

转载 作者:行者123 更新时间:2023-12-04 02:05:10 27 4
gpt4 key购买 nike

下面是如何在运行时使用 Manifest 创建类型 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 重新实现。_

下面是类(class) Test按照 som-snytt 的建议重新实现:
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]
}
}

只要我打电话 createInstanceTest有用:
val test = new Test[MyClass1]
val myClass = test.createInstance("hello") // this works

但是一旦我定义了一个派生自 Test 的新类...
class DerivedTest[T <: MyTrait] extends Test[T]

...然后我调用 createInstance在这样的新类(class)上......
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)
...

我错过了什么吗?

最佳答案

最终这里是实际工作的实现 - 请假设这个答案来自 som-snytt :

import scala.reflect.runtime._
import scala.reflect.runtime.universe._

class Test[T <: MyTrait : TypeTag] {

def createInstance(args: AnyRef*)(ctor: Int = 0): T = {
val tt = typeTag[T]

currentMirror.reflectClass(tt.tpe.typeSymbol.asClass).reflectConstructor(
tt.tpe.members.filter(m =>
m.isMethod && m.asMethod.isConstructor
).iterator.toSeq(ctor).asMethod
)(args: _*).asInstanceOf[T]
}
}

我希望这有帮助。

关于scala - 如何在运行时使用 TypeTags 创建类型 T 的实例,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21715164/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com