Reflection 使用反射调用Scala构造函数的参数数目错误

Reflection 使用反射调用Scala构造函数的参数数目错误,reflection,scala,scala-2.8,Reflection,Scala,Scala 2.8,我试图调用Scala类的构造函数的newInstance方法(case类或普通类都会受到影响) 但是,我遇到了一个带有错误参数数提示的IllegalArgumentException 考虑以下几点: case class Vec2(x: Float, y: Float) object TestApp { def main(args: Array[String]) { //after some research I found the last constructor always

我试图调用Scala类的构造函数的
newInstance
方法(case类或普通类都会受到影响)

但是,我遇到了一个带有错误参数数提示的
IllegalArgumentException

考虑以下几点:

case class Vec2(x: Float, y: Float)

object TestApp {
  def main(args: Array[String]) {
    //after some research I found the last constructor always to be the default
    val ctor = classOf[Vec2].getConstructors.last

    println("ctor = " + ctor)
    println("takes parameters: " + ctor.getParameterTypes.length)

    val params = new Array[Float](2)

    params.update(0, 1.0f)
    params.update(1, -1.0f)

    println("num parameters: " + params.length)

    println("trying to create new instance...")
    try {
      val x = ctor.newInstance(params)
      println("new instance: " + x)
    }
    catch {
      case ex => ex.printStackTrace
    }
  }
结果如下:

ctor = public pd.test.Vec2(float,float)
takes parameters: 2
num parameters: 2
trying to create new instance...
java.lang.IllegalArgumentException: wrong number of arguments
    at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
    at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:39)
    at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:27)
    at java.lang.reflect.Constructor.newInstance(Constructor.java:513)
    at pd.test.TestApp$.main(TestApp.scala:60)
    at pd.test.TestApp.main(TestApp.scala)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
    at java.lang.reflect.Method.invoke(Method.java:597)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:115)
我曾经在Java中经历过类似的事情。在这种情况下,我试图实例化的类是另一个类的内部类,因此Java需要一个隐式的附加参数,该参数要么是封闭类的
class
对象(如果该类声明为静态),要么是封闭类的实例

但是,在这种情况下,不存在
Vec2
的封闭类,除非Scala在内部添加一个(尽管,
java.lang.class.getEnclosuringClass()
为我返回
null


所以我的问题是如何使用反射实例化Scala类?Scala构造函数是否有隐式期望的其他参数?

newInstance方法采用varargs参数。在Scala中(与Java不同),您不能只传递一个数组并将其自动作为所有参数处理。如果这是您想要的,您需要明确地这样做:

ctor.newInstance(params:_*)

您现在正在做的是将一个包含2个元素的数组作为第一个参数传递给构造函数。

newInstance
除了
java.lang.Object
子类型的参数,因此您必须执行类似
ctor.newInstance(params.map(u.asInstanceOf[Object]):\u*
的操作,以便
Float
谢谢,那确实有用!也谢谢瓦西里。我正在处理的泛型类没有遇到这个问题,它创建了一个数组[AnyRef]。我发布的示例是一个简化版本。:)@Vasil与Java一样,Scala将自动装箱任何具有底层基元类型的内容,因此Scala
Float
可以是
Float
Java.lang.Float
。因为scala varargs使用
Seq
,而Seq是一种擦除类型,所以可以确定您传递的内容已经被装箱,并且将子类化
对象
@Kevin是的,我同意
Seq
。我记住了示例中使用的
Array
类型谢谢,这正是我想要的答案!