使用构造函数参数使用反射实例化对象

jus*_*tin 9 reflection scala

我试图弄清楚如何用反射实例化一个案例类对象.这有什么支持吗?我最接近的是查看scala.reflect.Invocation,但这似乎更适合执行作为对象一部分的方法.

case class MyClass(id:Long, name:String)

def instantiate[T](className:String)(args:Any*) : T = { //your code here }
Run Code Online (Sandbox Code Playgroud)

接近我正在寻找的API.

任何帮助,将不胜感激.

mic*_*ebe 19

scala> case class Foo(id:Long, name:String)
defined class Foo

scala> val constructor = classOf[Foo].getConstructors()(0)
constructor: java.lang.reflect.Constructor[_] = public Foo(long,java.lang.String)

scala> val args = Array[AnyRef](new java.lang.Integer(1), "Foobar")
args: Array[AnyRef] = Array(1, Foobar)

scala> val instance = constructor.newInstance(args:_*).asInstanceOf[Foo]
instance: Foo = Foo(1,Foobar)

scala> instance.id
res12: Long = 1

scala> instance.name
res13: String = Foobar

scala> instance.getClass
res14: java.lang.Class[_] = class Foo
Run Code Online (Sandbox Code Playgroud)

目前Scala没有太多的反射支持.但是你可以回到Java Reflection API.但是有一些障碍:

  • 你必须创建一个Array[AnyRef]并在包装类中包装你的"原始类型"(java.lang.Integer, java.lang.Character, java.lang.Double, ...)

  • newInstance(Object ... args)得到一个varargs数组Object,所以你应该给类型提示一个提示:_*

  • newInstance(...)返回一个,Object所以你必须把它再回来asInstanceOf[T]

我能得到的最接近你的instantiate功能是这样的:

def instantiate(clazz: java.lang.Class[_])(args:AnyRef*): AnyRef = {
  val constructor = clazz.getConstructors()(0)
  return constructor.newInstance(args:_*).asInstanceOf[AnyRef]
}

val instance = instantiate(classOf[MyClass])(new java.lang.Integer(42), "foo")
println(instance)           // prints: MyClass(42,foo)
println(instance.getClass)  // prints: class MyClass
Run Code Online (Sandbox Code Playgroud)

您无法从泛型类型中获取get类.Java会删除它(类型擦除).

编辑:2012年9月20日

三年后,instantiate可以改进该方法以返回正确类型的对象.

def instantiate[T](clazz: java.lang.Class[T])(args:AnyRef*): T = {
  val constructor = clazz.getConstructors()(0)
  return constructor.newInstance(args:_*).asInstanceOf[T]
}
Run Code Online (Sandbox Code Playgroud)

http://www.nabble.com/How-do-I-get-the-class-of-a-Generic--td20873455.html