在运行时将文件添加到 java 类路径

是否可以在运行时向 java 类路径添加文件(不一定是 jar 文件)。 具体来说,该文件已经存在于类路径中,我想要的是能否将该文件的修改后的副本添加到类路径中。

谢谢,

113708 次浏览

你可以的。如果您想要隔离它,那么它将需要位于其包结构中一个独立于编译代码其余部分的目录中。然后将其基目录放在命令行类路径的前面。

是的,我相信这是可能的,但是您可能必须实现您自己的类加载器。我从来没有这样做过,但这是我可能会看到的道路。

您可以尝试将 java.net.URLClassloader 与您更新的类所在的文件夹/jar 的 URL 一起使用,并在创建新线程时使用它而不是默认的类加载器。

只能将文件夹或 jar 文件添加到类装入器。因此,如果只有一个类文件,则需要首先将其放入适当的文件夹结构中。

这里的 是一个在运行时添加到 SystemClassLoader 的相当丑陋的代码:

import java.io.IOException;
import java.io.File;
import java.net.URLClassLoader;
import java.net.URL;
import java.lang.reflect.Method;


public class ClassPathHacker {


private static final Class[] parameters = new Class[]{URL.class};


public static void addFile(String s) throws IOException {
File f = new File(s);
addFile(f);
}//end method


public static void addFile(File f) throws IOException {
addURL(f.toURL());
}//end method




public static void addURL(URL u) throws IOException {


URLClassLoader sysloader = (URLClassLoader) ClassLoader.getSystemClassLoader();
Class sysclass = URLClassLoader.class;


try {
Method method = sysclass.getDeclaredMethod("addURL", parameters);
method.setAccessible(true);
method.invoke(sysloader, new Object[]{u});
} catch (Throwable t) {
t.printStackTrace();
throw new IOException("Error, could not add URL to system classloader");
}//end try catch


}//end method


}//end class

反射是访问受保护的方法 addURL所必需的。如果有 SecurityManager,这可能会失败。

试试这个。

private static void addSoftwareLibrary(File file) throws Exception {
Method method = URLClassLoader.class.getDeclaredMethod("addURL", new Class[]{URL.class});
method.setAccessible(true);
method.invoke(ClassLoader.getSystemClassLoader(), new Object[]{file.toURI().toURL()});
}

这将编辑系统类装入器以包含给定的库 jar。

我这样做的方法是使用我自己的类装入器

URLClassLoader urlClassLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
DynamicURLClassLoader dynalLoader = new DynamicURLClassLoader(urlClassLoader);

然后创建以下类:

public class DynamicURLClassLoader extends URLClassLoader {


public DynamicURLClassLoader(URLClassLoader classLoader) {
super(classLoader.getURLs());
}


@Override
public void addURL(URL url) {
super.addURL(url);
}
}

没有任何反射

我的解决办法是:

File jarToAdd = new File("/path/to/file");


new URLClassLoader(((URLClassLoader) ClassLoader.getSystemClassLoader()).getURLs()) {
@Override
public void addURL(URL url) {
super.addURL(url);
}
}.addURL(jarToAdd.toURI().toURL());