我有需要重写的createOld
方法,我无法更改它。我想使用TypeTag
模式匹配createNew
中提供的类型。目的是找出如何从createNew
调用createOld
。我当前的理解是,如果A
尚未附带createOld
方法中的TypeTag[A]
类型,则编译器没有足够的类型信息。
object TypeTagFromClass {
class C1
class C2
// How to get TypeTag[A] needed by createNew?
def createOld[A](c: Class[A]): A = createNew ???
def createNew[A : TypeTag]: A = {
val result = typeOf[A] match {
case a if a =:= typeOf[C1] => new C1()
case a if a =:= typeOf[C2] => new C2()
}
result.asInstanceOf[A]
}
}
最佳答案
可以使用Scala反射从TypeTag
创建Class
,尽管我不确定TypeCreator
的这种实现是否绝对正确:
import scala.reflect.runtime.universe._
def createOld[A](c: Class[A]): A = createNew {
val mirror = runtimeMirror(c.getClassLoader) // obtain runtime mirror
val sym = mirror.staticClass(c.getName) // obtain class symbol for `c`
val tpe = sym.selfType // obtain type object for `c`
// create a type tag which contains above type object
TypeTag(mirror, new TypeCreator {
def apply[U <: Universe with Singleton](m: api.Mirror[U]) =
if (m eq mirror) tpe.asInstanceOf[U # Type]
else throw new IllegalArgumentException(s"Type tag defined in $mirror cannot be migrated to other mirrors.")
})
}
但是,如果您不需要检查通用参数和完整的Scala类型信息,则实际上并不需要完整的
TypeTag
。您可以为此使用ClassTag
:def createNew[A: ClassTag]: A = {
val result = classTag[A].runtimeClass match {
case a if a.isAssignableFrom(classOf[C1]) => new C1()
case a if a.isAssignableFrom(classOf[C2]) => new C2()
}
result.asInstanceOf[A]
}
或带有一些隐含的糖:
implicit class ClassTagOps[T](val classTag: ClassTag[T]) extends AnyVal {
def <<:(other: ClassTag[_]) = classTag.runtimeClass.isAssignableFrom(other.runtimeClass)
}
def createNew[A: ClassTag]: A = {
val result = classTag[A] match {
case a if a <<: classTag[C1] => new C1()
case a if a <<: classTag[C2] => new C2()
}
result.asInstanceOf[A]
}
您可以使用普通的旧Java
newInstance()
方法进一步简化该操作:def createNew[A: ClassTag]: A = classTag[A].runtimeClass.newInstance().asInstanceOf[A]
当然,这仅在不需要为不同类使用不同构造函数参数的情况下才有效。
从
createNew
调用此createOld
比使用TypeTag
的调用简单得多:def createOld[A](c: Class[A]): A = createNew(ClassTag[A](c))
关于scala - 从类[A]获取TypeTag [A],我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/22970209/