如何在运行时加载 jar 文件

我被要求构建一个 Java 系统,它能够在运行时加载新的代码(扩展)。 在代码运行时如何重新加载 jar 文件? 或者如何加载新的 jar?

显然,由于恒定的正常运行时间很重要,因此我想增加在正常运行时重新加载现有类的能力(如果这不会使事情变得太复杂的话)。

我应该注意什么? (可以把它看作是两个不同的问题——一个是关于在运行时重新加载类的问题,另一个是关于添加新类的问题)。

159716 次浏览

我谷歌了一下,发现了这个代码 给你:

File file = getJarFileToLoadFrom();
String lcStr = getNameOfClassToLoad();
URL jarfile = new URL("jar", "","file:" + file.getAbsolutePath()+"!/");
URLClassLoader cl = URLClassLoader.newInstance(new URL[] {jarfile });
Class loadedClass = cl.loadClass(lcStr);

任何人可以分享关于这种方法的意见/评论/答案吗?

用现有的数据重新加载现有的类可能会造成破坏。

您可以相对容易地将新代码加载到新的类加载器中:

ClassLoader loader = URLClassLoader.newInstance(
new URL[] { yourURL },
getClass().getClassLoader()
);
Class<?> clazz = Class.forName("mypackage.MyClass", true, loader);
Class<? extends Runnable> runClass = clazz.asSubclass(Runnable.class);
// Avoid Class.newInstance, for it is evil.
Constructor<? extends Runnable> ctor = runClass.getConstructor();
Runnable doRun = ctor.newInstance();
doRun.run();

可以对不再使用的类加载器进行垃圾回收(除非存在内存泄漏,使用 ThreadLocal、 JDBC 驱动程序、 java.beans等常常会出现这种情况)。

如果您希望保留对象数据,那么我建议使用一种持久性机制,比如 Serialization,或者任何您习惯使用的机制。

当然,调试系统可以做一些更花哨的事情,但是它们更不可靠。

可以将新类添加到类装入器中。例如,使用 URLClassLoader.addURL。但是,如果一个类无法加载(比如说,因为您没有添加它) ,那么它将永远无法在该类加载器实例中加载。

我被要求建立一个 Java 系统,将有能力加载新的代码,同时运行

您可能希望将系统建立在 OSGi基础之上(或者至少在这方面花费很多精力) ,而 OSGi正是为这种情况而设计的。

扰乱类加载器确实是一件棘手的事情,主要是因为类可见性是如何工作的,而且您不希望以后遇到难以调试的问题。例如,在许多库中广泛使用的 ForName ()在支离破碎的类加载器空间中不能很好地工作。

这对我有用:

File file  = new File("c:\\myjar.jar");


URL url = file.toURL();
URL[] urls = new URL[]{url};


ClassLoader cl = new URLClassLoader(urls);
Class cls = cl.loadClass("com.mypackage.myclass");

使用 org.openide.util.Lookup 和 ClassLoader 动态加载 Jar 插件,如下所示。

public LoadEngine() {
Lookup ocrengineLookup;
Collection<OCREngine> ocrengines;
Template ocrengineTemplate;
Result ocrengineResults;
try {
//ocrengineLookup = Lookup.getDefault(); this only load OCREngine in classpath of  application
ocrengineLookup = Lookups.metaInfServices(getClassLoaderForExtraModule());//this load the OCREngine in the extra module as well
ocrengineTemplate = new Template(OCREngine.class);
ocrengineResults = ocrengineLookup.lookup(ocrengineTemplate);
ocrengines = ocrengineResults.allInstances();//all OCREngines must implement the defined interface in OCREngine. Reference to guideline of implement org.openide.util.Lookup for more information


} catch (Exception ex) {
}
}


public ClassLoader getClassLoaderForExtraModule() throws IOException {


List<URL> urls = new ArrayList<URL>(5);
//foreach( filepath: external file *.JAR) with each external file *.JAR, do as follows
File jar = new File(filepath);
JarFile jf = new JarFile(jar);
urls.add(jar.toURI().toURL());
Manifest mf = jf.getManifest(); // If the jar has a class-path in it's manifest add it's entries
if (mf
!= null) {
String cp =
mf.getMainAttributes().getValue("class-path");
if (cp
!= null) {
for (String cpe : cp.split("\\s+")) {
File lib =
new File(jar.getParentFile(), cpe);
urls.add(lib.toURI().toURL());
}
}
}
ClassLoader cl = ClassLoader.getSystemClassLoader();
if (urls.size() > 0) {
cl = new URLClassLoader(urls.toArray(new URL[urls.size()]), ClassLoader.getSystemClassLoader());
}
return cl;
}