getResourceAsStream没有在webapp中加载资源
我有一个Web应用程序,它使用驻留在TOMCAT_HOME / common / lib中的库。 此库在类路径的根(在名为ApplicationConfig的类中)中查找属性文件:
ApplicationConfig.class.getResourceAsStream("/hv-application.properties");
我的Tomcat Web应用程序包含此属性文件。 它在WEB-INF / classes中,它是classpath的根本吗? 但是,在运行时,当它尝试加载属性文件时,它会抛出exception,因为它无法找到它(getResourceAsStream返回null)。
如果我的应用程序是一个简单的独立Java应用程序,那么一切正常。 Tomcat会导致getResourceAsStream方法采取不同的行为吗? 我知道那里有很多类似的问题,但不幸的是他们都没有帮助过。 谢谢。
请尝试使用Thread.currentThread().getContextClassLoader().getResourceAsStream("/hv-application.properties")
。
这看起来可能与Tomcat类加载器的工作方式有关。 如果你在一个类加载器(webapp类加载器中的配置文件)中有某些东西被另一个(jar in common / lib)中的东西使用,那么结果可能会非常令人头疼。
本文档解释了Tomcat如何委托类加载器。 如果可能,您可以尝试以下之一:
- 将common / lib中的jar文件移动到Web应用程序(WEB-INF / lib)中。 我知道这并不总是可行,但有时jars(例如log4j)可以跨类加载器(*)和平共存。
- 将配置文件移动到common / classes中。 这实际上是相同的(将配置项放入与需要它的jar相同的类加载器中)。 同样,这不是理想的,但如果您可以控制您的环境,它就会起作用。
无论哪种方式,在不同的类加载器中拥有资源可能会很痛苦。 我希望这有帮助。
(*)log4j具有-log4j.ignoreTCL
选项,这使得这成为可能
Tomcat安全管理器通常不允许您从Tomcat根库中的库访问webapp类和资源。 这是为了在容器中运行的Web应用程序之间进行分离。
您应该能够通过更新安全策略来解决这个问题,但通常最好不要将您的lib放入Tomcat容器中,我假设您正在这样做。
我正在扩大奥利维尔的评论作为回应(谢谢你的领导)。
问题似乎是资源路径中的前导斜杠(/)。
在Tomcat 8中 ,WebAppClassloader正确地解析了带有和不带前导斜杠的路径。 两个.getResourceAsStream("/org/pakopa/app/config.properties");
和.getResourceAsStream("org/pakopa/app/config.properties");
返回一个InputStream。
在Tomcat 7中 (我也假设以前的版本) .getResourceAsStream("/org/pakopa/app/config.properties");
未解析并返回null
但.getResourceAsStream("org/pakopa/app/config.properties");
正确解决了。