12

我想要一个方法

def retrieve[T](value: Option[T])(implicit ct: ClassTag[T]): T;

在这个方法中,我需要调用一个 Java 方法(超出我的控制范围)来创建一个T需要的实例Class[T]

public <T> T construct(clazz: Class<T> /* other arguments */) { ... }

我怎样才能Class[T]得到ClassTag[T]?首先我以为我可以使用runtimeClassfrom ClassTag,但它的类型是Class[_],不是Class[T]。还是编译器可以自动提供任何其他隐式值,我可以从中获得Class[T]

4

1 回答 1

14

这是奥德斯基推测的门票getClass链接的论坛讨论:

你也可以使用演员表。

这是固定的重复票getClass5.getClass还投:

/** Return the class object representing an unboxed value type,
 *  e.g. classOf[int], not classOf[java.lang.Integer].  The compiler
 *  rewrites expressions like 5.getClass to come here.
 */
def anyValClass[T <: AnyVal : ClassTag](value: T): jClass[T] =
  classTag[T].runtimeClass.asInstanceOf[jClass[T]]

这个限制让人想起这个关于模式匹配的问题ClassTag,我们的幼稚期望也没有得到满足。

电阻是否Class[A]代表 Scala 类型和平台之间的阻抗不匹配?

给定类类型,真正能做的就是newInstance. 但是使用构造函数镜像的反射调用不会给我我的类型。

scala> res24 reflectConstructor res25.asMethod
res27: reflect.runtime.universe.MethodMirror = constructor mirror for Bar.<init>(): Bar (bound to null)

scala> res27()
res28: Any = Bar@2eeb08d9

scala> bar.getClass.newInstance
res29: Bar = Bar@31512f0a

scala> classOf[Bar].newInstance
res30: Bar = Bar@2bc1d89f

这似乎不公平。

正如 2008 年的邮件主题所总结的那样,您希望在 Scala 中使用更少的强制转换。

顺便说一句,并不是我不相信代码注释,而是:

scala> 5.getClass
res38: Class[Int] = int

scala> :javap -
  Size 1285 bytes
  MD5 checksum a30a28543087238b563fb1983d7d139b
  Compiled from "<console>"

[剪辑]

 9: getstatic     #27                 // Field scala/runtime/ScalaRunTime$.MODULE$:Lscala/runtime/ScalaRunTime$;
12: iconst_5      
13: invokestatic  #33                 // Method scala/runtime/BoxesRunTime.boxToInteger:(I)Ljava/lang/Integer;
16: getstatic     #38                 // Field scala/reflect/ClassTag$.MODULE$:Lscala/reflect/ClassTag$;
19: invokevirtual #42                 // Method scala/reflect/ClassTag$.Int:()Lscala/reflect/ClassTag;
22: invokevirtual #46                 // Method scala/runtime/ScalaRunTime$.anyValClass:(Ljava/lang/Object;Lscala/reflect/ClassTag;)Ljava/lang/Class;
25: putfield      #18                 // Field res38:Ljava/lang/Class;
28: return        
于 2013-08-05T00:03:56.767 回答