mos*_*awi 1 java reflection classloader
我试图jars
在顶层调用同一依赖库的2个不同版本Main class
。因此,我创建了一个具有2个实现类的接口,这两个类都有一个run方法,该方法使用一个将要使用的通用api somejar-1.0.0-SNAPSHOT.jar
,另一个将somejar-2.0.0-SNAPSHOT.jar
通过显式调用ClassLoader 来使用。
public static void main(String[] args) throws MalformedURLException, ClassNotFoundException, InstantiationException, IllegalAccessException {
ClassLoader loader1 = new URLClassLoader( new URL[] { new File("/Users/haddad/.m2/repository/com/company/somejar-1.0.0-SNAPSHOT.jar").toURL() });
ClassLoader loader2 = new URLClassLoader( new URL[] { new File("/Users/haddad/.m2/repository/com/company/somejar-2.0.0-SNAPSHOT.jar").toURL() });
Class<?> c1 = loader1.loadClass("com.engine.na.EngineV1");
Class<?> c2 = loader2.loadClass("com.engine.na.EngineV2");
IEngine app1 = (IEngine) c1.newInstance();
IEngine app2 = (IEngine) c2.newInstance();
Integer s1 = app1.run();
Integer s2 = app2.run();
Assert.equals(s1,s2,"Outputs from somejar-1.0 and somejar-2.0 did not match, perhaps somejar-2.0 has regressed?");
}
Run Code Online (Sandbox Code Playgroud)
这是带有接口的EngineV1和V2:
public Interface IEngine {
Integer run();
}
public class EngineV1 implements IEngine {
private File content;
private File en;
public EngineV1(args) {
this.content = new File("/some/path");
this.en = new File("/some/path");
}
public static void main(String[] args) {
new EngineV1(args).run();
}
public Integer run() {
// some logic...
somejar.evaluateSpeed();
}
}
public class EngineV2 implements IEngine {
private File content;
private File en;
public EngineV2(args) {
this.content = new File("/some/path");
this.en = new File("/some/path");
}
public static void main(String[] args) {
new EngineV2(args).run();
}
public Integer run() {
// some logic...
somejar.evaluateSpeed();
}
}
Run Code Online (Sandbox Code Playgroud)
当我去上主课时,我得到:
Exception in thread "main" java.lang.InstantiationException: com.engine.na.EngineV1
at java.lang.Class.newInstance(Class.java:427)
at com.engine.na.MainClass.main(MainClass.java:23)
Caused by: java.lang.NoSuchMethodException: com.engine.na.EngineV1.<init>()
at java.lang.Class.getConstructor0(Class.java:3082)
at java.lang.Class.newInstance(Class.java:412)
... 1 more
Run Code Online (Sandbox Code Playgroud)
为什么会出现此错误?如何解决呢?
在类中声明构造函数时,不带参数的默认构造函数不会在编译时生成。
您有两种方法:
显式添加不带参数的构造函数。
通过反射调用带参数的构造函数。
第一种方法是简单的技巧,是一种非常通用的方法,但是如果在创建状态时重视状态或实例的一部分时,它不一定是最佳方法。
对于第一种情况,这是不言自明的。
对于第二种情况,想法是您必须从类中检索带有参数的构造函数,并在使用它时指定参数。
例如,使用以下构造函数:
public EngineV1(String value) {
...
}
Run Code Online (Sandbox Code Playgroud)
您可以通过以下方式调用它:
Class<EngineV1> c1 = (Class<EngineV1>)loader1.loadClass("com.engine.na.EngineV1");
Constructor<EngineV1> constructor = c1.getConstructor(String.class);
EngineV1 instance = ctor.newInstance("myString");
Run Code Online (Sandbox Code Playgroud)