Von*_*onC 78
2017年第四季度更新:如下面由vda8888 评论,在Java 9中,系统不再是一个.java.lang.ClassLoader
java.net.URLClassLoader
请参阅" Java 9迁移指南:七个最常见的挑战 "
我刚刚描述的类加载策略是以新类型实现的,而在Java 9中,应用程序类加载器属于该类型.
这意味着它URLClassLoader
不再是,所以偶尔(URLClassLoader) getClass().getClassLoader()
或(URLClassLoader) ClassLoader.getSystemClassLoader()
序列将不再执行.
java.lang.ModuleLayer将是一种替代方法,用于影响模块路径(而不是类路径).请参阅" Java 9模块 - JPMS基础知识 ".
对于Java 8或更低版本:
一些一般性意见:
你不能(以可移植的方式保证工作,见下文)改变系统类路径.相反,您需要定义一个新的ClassLoader.
ClassLoaders以分层方式工作...因此,任何对类X进行静态引用的类都需要在与X相同的ClassLoader中加载,或者在子类ClassLoader中加载.您不能使用任何自定义ClassLoader来正确地使系统ClassLoader链接加载代码,如果它之前没有这样做的话.因此,除了您找到的额外代码之外,您还需要安排主应用程序代码在自定义ClassLoader中运行.
(话虽如此,破解 - 所有人都在评论中提到这个扩展的 URLClassLoader
例子)
你可能会考虑不编写自己的ClassLoader,而只是使用URLClassLoader.使用不在父类加载器URL中的URL创建URLClassLoader .
URL[] url={new URL("file://foo")};
URLClassLoader loader = new URLClassLoader(url);
Run Code Online (Sandbox Code Playgroud)
一个更完整的解决方案将是:
ClassLoader currentThreadClassLoader
= Thread.currentThread().getContextClassLoader();
// Add the conf dir to the classpath
// Chain the current thread classloader
URLClassLoader urlClassLoader
= new URLClassLoader(new URL[]{new File("mtFile").toURL()},
currentThreadClassLoader);
// Replace the thread classloader - assumes
// you have permissions to do so
Thread.currentThread().setContextClassLoader(urlClassLoader);
Run Code Online (Sandbox Code Playgroud)
如果您假设JVMs系统类加载器是URLClassLoader(对于所有JVM可能都不是这样),您也可以使用反射来实际修改系统类路径...(但这是一个hack;)):
public void addURL(URL url) throws Exception {
URLClassLoader classLoader
= (URLClassLoader) ClassLoader.getSystemClassLoader();
Class clazz= URLClassLoader.class;
// Use reflection
Method method= clazz.getDeclaredMethod("addURL", new Class[] { URL.class });
method.setAccessible(true);
method.invoke(classLoader, new Object[] { url });
}
addURL(new File("conf").toURL());
// This should work now!
Thread.currentThread().getContextClassLoader().getResourceAsStream("context.xml");
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
70036 次 |
最近记录: |