Java - 如何加载同一类的不同版本?

2022-09-01 01:03:44

我已经阅读了很多关于Java类装入器的文章,但到目前为止,我还没有找到这个简单问题的答案:

我有两个版本的com.abc.Hello.class在jars v1.jarv2.jar中。我想在我的应用程序中同时使用这两者。最简单的方法是什么?

我不期望那么简单,但是沿着这些路线的东西会很棒:

Classloader myClassLoader = [magic that includes v1.jar and ignores v2.jar]
Hello hello = myclassLoader.load[com.abc.Hello]

在另一个班级:

Classloader myClassLoader = [magic that includes v2.jar and ignores v1.jar]
Hello hello = myclassLoader.load[com.abc.Hello]

我想避免使用OSGi。


答案 1

你走的路是正确的。你必须考虑一些事情。

正常的情况是使用父类装入器中存在的类。因此,如果您想要两个版本,则这些类一定不存在。

但是,如果你想进行交互,你可以使用反射,或者更好的是使用通用接口。所以我会这样做:

common.jar:
BaseInterface

v1.jar:
SomeImplementation implements BaseInterface

v2.jar:
OtherImplementation implements BaseInterface

command-line:
java -classpath common.jar YourMainClass
// you don't put v1 nor v2 into the parent classloader classpath

Then in your program:

loader1 = new URLClassLoader(new URL[] {new File("v1.jar").toURL()}, Thread.currentThread().getContextClassLoader());
loader2 = new URLClassLoader(new URL[] {new File("v2.jar").toURL()}, Thread.currentThread().getContextClassLoader());

Class<?> c1 = loader1.loadClass("com.abc.Hello");
Class<?> c2 = loader2.loadClass("com.abc.Hello");

BaseInterface i1 = (BaseInterface) c1.newInstance();
BaseInterface i2 = (BaseInterface) c2.newInstance();

答案 2

您几乎已经编写了解决方案。我希望下面的代码片段会有所帮助。

ClassLoader cl = new URLClassLoader(new URL[] {new File("v1.jar").toURL()}, Thread.currentThread().getContextClassLoader());
Class<?> clazz = cl.loadClass("Hello");

替换为,此代码将加载版本 #2。v1.jarv2.jar