Reflection 使用反射调用Scala构造函数的参数数目错误
我试图调用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
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将自动装箱任何具有底层基元类型的内容,因此ScalaFloat
可以是Float
或Java.lang.Float
。因为scala varargs使用Seq
,而Seq是一种擦除类型,所以可以确定您传递的内容已经被装箱,并且将子类化对象
@Kevin是的,我同意Seq
。我记住了示例中使用的Array
类型谢谢,这正是我想要的答案!