我已经在运行时编译了一个类,我想动态使用它,只要它的构造函数有一个参数
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
代替,IllegalArgumentException
调用期间newInstance
只让我更进了一步(当然这个.manager是com.mycompany.Manager)
Constructor<?> list[] = clazz.getConstructors();
Constructor<?> constructor = list[0];
this.customProcessor = (Processor) constructor.newInstance( this.manager);
我做的Watever,运行时管理器对象与编译之间存在不匹配
如何修复此构造函数签名?
修改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();
这一切似乎都是同一问题的一部分,其中运行时类似乎与编译不一致,尽管名称匹配。
答案 0 :(得分:1)
您的CustomProcessor类没有构造函数,因为您认为是构造函数的方法的名称是不同的。
public CustomLatencyProcessor(Manager m) {
super(m);
}
应改为
public CustomProcessor(Manager m) {
super(m);
}
因为您的类的名称是CustomProcessor。构造函数的名称必须与其包含类的名称完全匹配。
答案 1 :(得分:1)
在使用使用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");
我希望这可以为你节省2天!