2

我有一个 DSL 和一些运行时代码。问题是我在运行时到达某个地方:

val clazz: Class[_ <: java.io.Serializable] = classOf[java.lang.String]
val value: java.io.Serializable = "abc"

我有一堂课

class MyWrapper[T <: java.io.Serializable](val data: T)(implicit m: Manifest[T]) {
  override def toString = m.runtimeClass
}
val wrapper = new MyWrapper(value)

问题是我需要java.lang.StringtoString. 但我得到了java.io.Serializable。不幸的是,我既不能为每个java.io.Serializable子类型创建固定的模式匹配(这太疯狂了),也不MyWrapper能用new MyWrapper[String](value). 我不知道 的类型value,也许它是 的子类型Serializable

如果我知道值类型等于变量,是否有某种方法可以在运行时传递类型/清单值clazz


更新(Régis Jean-Gilles 的解决方案不起作用)。REPL测试:

val clazz: Class[_ <: java.io.Serializable] = classOf[java.lang.String]
val value: java.io.Serializable = "abc"

class MyWrapper[T <: java.io.Serializable](val data: T)(implicit m: Manifest[T]) {
  override def toString = m.runtimeClass.getName
}
val wrapper = new MyWrapper(value)
//val wrapper = new MyWrapper(value)(ClassManifest.fromClass(clazz).asInstanceOf[ClassTypeManifest[java.io.Serializable]])
//val wrapper = new MyWrapper(value)(ClassManifest.fromClass(clazz))
System.err.println(wrapper.toString)

如果我尝试显式传递清单,则会出现错误:

scala> :load test.scala
Loading test.scala...
clazz: Class[_ <: java.io.Serializable] = class java.lang.String
value: java.io.Serializable = abc
defined class MyWrapper
<console>:10: error: type mismatch;
 found   : scala.reflect.ClassManifest[_$1] where type _$1 <: java.io.Serializable
    (which expands to)  scala.reflect.ClassTag[_$1]
 required: Manifest[java.io.Serializable]
       val wrapper = new MyWrapper(value)(ClassManifest.fromClass(clazz))
                                                                 ^
<console>:8: error: not found: value wrapper
              System.err.println(wrapper.toString)

我也无法明确表达清单。

当我尝试编译我的应用程序时出现更奇怪的错误 -

[error]  found   : scala.reflect.ClassManifest[(some other)_0(in method setProperty)]
[error]     (which expands to)  scala.reflect.ClassTag[(some other)_0(in method setProperty)]
[error]  required: Manifest[_0(in method setProperty)]
[error]           new Value.Static(default, Context.virtual(element))(ClassManifest.fromClass(elementProperty.typeClass)))

恕我直言,Régis Jean-Gilles 非常接近解决方案。如何使其与 Scala 2.10 一起使用?

4

2 回答 2

2

如果我理解正确,您正在使用清单来解决类型擦除问题,但在特定点上,您所拥有的只是 a Class[_],因此您需要将其转换回清单。正确的?

如果是这样,您可以使用ClassManifest.fromClass, 然后将其作为隐式值显式传递m

val wrapper = new MyWrapper(value)(Manifest.classType(clazz))

您可能应该将其抽象出来:

object MyWrapper {
  def fromRuntimeType[T <: java.io.Serializable]( value: T ): MyWrapper[T] = {
    new MyWrapper(value)(Manifest.classType(value.getClass))
  }
}

但请记住,因为您首先只有一个 Class实例,所以您再次受到类型擦除的摆布。这意味着如果类(您需要为其获取清单)是泛型的,则在返回的清单中将没有任何类型参数信息ClassManifest.fromClass

于 2013-01-29T12:54:28.523 回答
0

不会

override def toString = data.getClass.getName

已经为你工作了吗?这会产生java.lang.String.

于 2013-01-29T12:58:06.473 回答