为了更好地理解 Java 中的工作原理,我想知道我是否可以在运行时动态添加目录到类路径。
例如,如果我使用 "java -jar mycp.jar" 启动 .jar 并输出 java.class.path 属性,我可能会得到:
java.class.path: '.:/Library/Java/Extensions:/System/Library/Java/Extensions:/usr/lib/java'
现在我可以在运行时修改这个类路径来添加另一个目录吗? (例如,在使用位于我要添加的目录中的 .jar 首次调用类之前)。
您可以使用以下方法:
URLClassLoader.addURL(URL url)
但是你需要通过反思来做到这一点,因为方法是
protected
:
public static void addPath(String s) throws Exception {
File f = new File(s);
URL u = f.toURL();
URLClassLoader urlClassLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
Class urlClass = URLClassLoader.class;
Method method = urlClass.getDeclaredMethod("addURL", new Class[]{URL.class});
method.setAccessible(true);
method.invoke(urlClassLoader, new Object[]{u});
}
参见 Reflection 上的 Java Trail。特别是反思的缺点
部分2014 年更新:这是 Jonathan Spooner 从 2011 年开始接受的答案中的代码,稍微重写以让 Eclipse 的验证器不再创建警告(弃用,原始类型)
//need to do add path to Classpath with reflection since the URLClassLoader.addURL(URL url) method is protected:
public static void addPath(String s) throws Exception {
File f = new File(s);
URI u = f.toURI();
URLClassLoader urlClassLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
Class<URLClassLoader> urlClass = URLClassLoader.class;
Method method = urlClass.getDeclaredMethod("addURL", new Class[]{URL.class});
method.setAccessible(true);
method.invoke(urlClassLoader, new Object[]{u.toURL()});
}
是的,您可以使用
URLClassLoader
.. 请参见示例 here。不使用反射。
-- 编辑 --
按照建议从链接复制示例。
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Hashtable;
import javax.naming.*;
public class ChangeLoader {
public static void main(String[] args) throws MalformedURLException {
if (args.length != 1) {
System.err.println("usage: java ChangeLoader extra_classpath_url");
System.exit(-1);
}
String url = args[0];
// Save class loader so that we can restore later
ClassLoader prevCl = Thread.currentThread().getContextClassLoader();
// Use prevCl as parent, so we extend instead of replace
ClassLoader urlCl =
URLClassLoader.newInstance(new URL[] {new URL(url)}, prevCl);
try {
Thread.currentThread().setContextClassLoader(urlCl);
Context ctx = new InitialContext();
System.out.println(ctx.lookup("tutorial/report.txt"));
ctx.close();
} catch (NamingException e) {
e.printStackTrace();
} finally {
// Restore old loader
Thread.currentThread().setContextClassLoader(prevCl);
}
} // main()
}