一尘不染

JavaClass.getResource()和ClassLoader.getResource()有什么区别?

java

我想知道Class.getResource()和之间有什么区别ClassLoader.getResource()

编辑:我特别想知道在文件/目录级别是否涉及任何缓存。如“目录列表是否缓存在Class版本中?”

AFAIK以下基本上应该做同样的事情,但事实并非如此:

getClass().getResource() 
getClass().getClassLoader().getResource()

我在摆弄一些报表生成代码时发现了这一点,该代码WEB-INF/classes/从该目录中的现有文件中创建了一个新文件。使用Class中的方法时,我可以使用来找到部署getClass().getResource()时存在的文件,但是当尝试获取新创建的文件时,我收到了一个空对象。浏览目录清楚地表明新文件在那里。文件名前面有一个正斜杠,如“ /myFile.txt”中所示。

另一方面,的ClassLoader版本getResource()确实找到了生成的文件。从这种经验看来,目录列表存在某种形式的缓存。我是对的,如果是这样,在哪里记录?

从API文档开始Class.getResource()

查找具有给定名称的资源。用于搜索与给定类关联的资源的规则由该类的定义类加载器实现。该方法委托给该对象的类加载器。如果此对象是由引导类加载器加载的,则该方法将委托给ClassLoader.getSystemResource(java.lang.String)。

对我来说,这是“ Class.getResource实际上正在调用其自己的类加载器的getResource()”。这和做一样getClass().getClassLoader().getResource()。但这显然不是。有人能为我提供一些启发吗?


阅读 716

收藏
2020-02-27

共2个答案

一尘不染

回答是否存在任何缓存的问题。

我通过运行一个独立的Java应用程序进一步研究了这一点,该应用程序使用getResourceAsStream ClassLoader方法从磁盘连续加载文件。我能够编辑该文件,并且更改会立即反映出来,即,该文件是从磁盘重新加载而无需缓存的。

但是: 我正在一个具有多个相互依赖的Maven模块和Web项目的项目中。我使用IntelliJ作为我的IDE来编译和运行Web项目。

我注意到上面的内容似乎不再成立,原因是我正在加载的文件现在被烘焙到jar中并部署到了相关的Web项目中。我只是在尝试更改目标文件夹中的文件后才注意到这一点,但无济于事。这使得好像正在进行缓存。

2020-02-27
一尘不染

Class.getResource可以采用“相对”资源名称,相对于类的包来对待。或者,您可以使用斜杠指定“绝对”资源名称。类加载器资源路径始终被认为是绝对的。

因此,以下基本上是等效的:

foo.bar.Baz.class.getResource("xyz.txt");
foo.bar.Baz.class.getClassLoader().getResource("foo/bar/xyz.txt");

这些也是(但它们与上面的有所不同):

foo.bar.Baz.class.getResource("/data/xyz.txt");
foo.bar.Baz.class.getClassLoader().getResource("data/xyz.txt");
2020-02-27