我在运行时编译了一个类,我想在运行时使用它,前提是它的构造函数接受一个参数
package com.notmycompany;
import com.mycompany.Manager;
import com.mycompany.Processor;
import com.mycompany.Event;
public class CustomProcessor extends Processor {
public CustomProcessor( Manager m) {
super( m);
}
@Override
public void process( Event evt) {
// Do you own stuff
System.out.println( "My Own Stuff");
}
}
编译进行得很好,我可以立即加载该类。但是建造师让我很难过。
Class<?> clazz = urlClassLoader.loadClass("com.notmycompany.CustomProcessor");
Constructor<?> constructor = clazz.getConstructor( com.mycompany.Manager.class);
this.customProcessor = (Processor) constructor.newInstance( this.manager);
在这种情况下,getConstructor
抛出一个NoSuchMethodException
我试着使用getConstructors
,这只会让我在newInstance
调用期间使用IllegalArgumentException
更进一步(当然,this.manager是com.mycompany.manager)
Constructor<?> list[] = clazz.getConstructors();
Constructor<?> constructor = list[0];
this.customProcessor = (Processor) constructor.newInstance( this.manager);
无论我做什么,运行时的Manager对象和编译之间都不匹配
如何修复此构造函数签名
编辑1:getParameterTypes输出
for( Class<?> c : constructor.getParameterTypes()) {
System.out.println( c);
}
输出
class com.mycompany.Manager
编辑2:我删除了构造函数参数作为临时解决方法
现在,当调用构造函数时,代码抛出ClassCastException
,抱怨com.notmycompany.CustomProcessor cannot be cast to com.mycompany.Processor
:
Constructor<?> constructor = clazz.getConstructor();
this.customProcessor = (Processor) constructor.newInstance();
这一切似乎都是同一个问题的一部分,尽管名称匹配,但运行时类似乎与编译不一致。
public CustomLatencyProcessor(Manager m) {
super(m);
}
应更改为
public CustomProcessor(Manager m) {
super(m);
}
因为类的名称是CustomProcessor。构造函数的名称必须与其包含类的名称完全匹配。
在使用一个使用currentThread作为父级的URL(而不是用URL从头开始创建的URLClassLoader)之后,我终于能够让它工作了
URLClassLoader ucl = (URLClassLoader)Thread.currentThread().getContextClassLoader();
URLClassLoader ucl2 = new URLClassLoader( new URL[] { new URL( "file://d:/temp/")},ucl);
Class<?> clazz = ucl2.loadClass("com.notmycompany.CustomProcessor");
我希望这能为你节省两天时间!