在Java
中,为了从相对路径读取文件,常常会使用的方法即是:html
xxx.class.getResource();
xxx.class.getClassLoader().getResource();
复制代码
在Spring
中,咱们还能够经过Spring
提供的Resource
进行一些操做:java
ClassPathResource
FileSystemResource
ServletContextResource
Resource template = ctx.getResource("some/resource/path/myTemplate.txt");
复制代码
这里简单总结下他们的区别:web
这个方法是今天的主角。spring
咱们都知道ClassLoader
的做用是用来加载.class
文件的,而且ClassLoader
是遵循Java
类加载中的双亲委派机制的。api
那么,ClassLoader
是如何找到这个.class
文件的呢?答案是URLClassPath
微信
Java
中自带了3个ClassLoader
分别是BootStrap ClassLoader
,EtxClassLoader
,AppClassLoader
,app
这3个ClassLoader
都继承自URLClassLoader
,而URLClassLoader
中包含一个URLClassPath
用来记录每一个ClassLoader
对应的加载.class
文件的路径,当须要加载资源的时候,只管从URLClassPath
对应的路径查找便可。webapp
下面是测试代码:ide
System.out.println("BootStrap ClassLoader ");
Stream.of(System.getProperty("sun.boot.class.path").split(";")).forEach(System.out::println);
System.out.println("ExtClassLoader:");
Stream.of(System.getProperty("java.ext.dirs").split(";")).forEach(System.out::println);
System.out.println("AppClassLoader:");
Stream.of(System.getProperty("java.class.path").split(";")).forEach(System.out::println);
复制代码
输出以下:测试
BootStrap ClassLoader
H:\java\jdk1.8\jre\lib\resources.jar
H:\java\jdk1.8\jre\lib\rt.jar
H:\java\jdk1.8\jre\lib\sunrsasign.jar
H:\java\jdk1.8\jre\lib\jsse.jar
H:\java\jdk1.8\jre\lib\jce.jar
H:\java\jdk1.8\jre\lib\charsets.jar
H:\java\jdk1.8\jre\lib\jfr.jar
H:\java\jdk1.8\jre\classes
ExtClassLoader:
H:\java\jdk1.8\jre\lib\ext
C:\Windows\Sun\Java\lib\ext
AppClassLoader:
H:\java\jdk1.8\jre\lib\charsets.jar
H:\java\jdk1.8\jre\lib\deploy.jar
H:\java\jdk1.8\jre\lib\ext\access-bridge-64.jar
H:\java\jdk1.8\jre\lib\ext\cldrdata.jar
H:\java\jdk1.8\jre\lib\ext\dnsns.jar
H:\java\jdk1.8\jre\lib\ext\jaccess.jar
H:\java\jdk1.8\jre\lib\ext\jfxrt.jar
H:\java\jdk1.8\jre\lib\ext\localedata.jar
H:\java\jdk1.8\jre\lib\ext\nashorn.jar
H:\java\jdk1.8\jre\lib\ext\sunec.jar
H:\java\jdk1.8\jre\lib\ext\sunjce_provider.jar
H:\java\jdk1.8\jre\lib\ext\sunmscapi.jar
H:\java\jdk1.8\jre\lib\ext\sunpkcs11.jar
H:\java\jdk1.8\jre\lib\ext\zipfs.jar
H:\java\jdk1.8\jre\lib\javaws.jar
H:\java\jdk1.8\jre\lib\jce.jar
H:\java\jdk1.8\jre\lib\jfr.jar
H:\java\jdk1.8\jre\lib\jfxswt.jar
H:\java\jdk1.8\jre\lib\jsse.jar
H:\java\jdk1.8\jre\lib\management-agent.jar
H:\java\jdk1.8\jre\lib\plugin.jar
H:\java\jdk1.8\jre\lib\resources.jar
H:\java\jdk1.8\jre\lib\rt.jar
F:\spring-test\target\classes
复制代码
AppClassLoader
负责经常使用的JDK jar
以及项目所依赖的jar
包上述参数能够经过 sun.misc.Launcher.class得到
经过输出的参数,咱们能够清晰的看出来各个
ClassLoader
负责的区域
说了这么多,这个和ClassLoader#getResource()
有什么关系呢?
关系很大,前面刚刚提问过,ClassLoader
是如何读取.class
文件的呢?
答案是URLClassPath#getResource()
方法:每一个UrlClassLoader
都是经过URLClassPath
来存储对应的加载区域,当须要查找.class
文件的时候,就经过URLClassPath#getResource()
查找便可。
下面再来看看ClassLoader#getResource()
//双亲委派查找
public URL getResource(String name) {
URL url;
if (parent != null) {
url = parent.getResource(name);
} else {
url = getBootstrapResource(name);
}
if (url == null) {
url = findResource(name);
}
return url;
}
//因为BootStrap ClassLoader是C++写的,Java拿不到其引用。
//所以这里单独写了一个方法获取BootStrapResource()
private static URL getBootstrapResource(String name) {
URLClassPath ucp = getBootstrapClassPath();
Resource res = ucp.getResource(name);
return res != null ? res.getURL() : null;
}
复制代码
URLClassLoader#findResource()
public URL findResource(final String name) {
URL url = AccessController.doPrivileged(
new PrivilegedAction<URL>() {
public URL run() {
return ucp.findResource(name, true);
}
}, acc);
return url != null ? ucp.checkURL(url) : null;
}
复制代码
咱们只用注意这一句ucp.findResource(name, true);
,这边是查找.class
文件的方法,所以咱们能够总结出经过ClassLoader#getResource()
的流程:
AppClassLoader
委派给ExtClassLoader
查找是否存在对应的资源ExtClassLoader
委派给BootStrap ClassLoader
查找是有存在对应的资源BootStrap ClassLoader
经过URLClasspath
查找本身加载的区域,查找到了即返回BootStrap ClassLoader
未查找到对应资源,ExtClassLoader
经过URLClasspath
查找本身加载的区域,查找到了即返回ExtClassLoader
未查找到对应资源,AppClassLoader
经过URLClasspath
查找本身加载的区域,查找到了即返回AppClassLoader
未查找到,抛出异常。这个过程,就和加载.class
文件的过程同样。
在这里咱们就能够发现,经过ClassLoader#getResource()
能够获取JDK
资源,所依赖的JAR
包资源等
所以,咱们甚至能够这样写:
//读取java.lang.String.class
的字节码
InputStream inputStream =Test.class.getClassLoader().getResourceAsStream("java/lang/String.class");
try(BufferedInputStream bufferedInputStream=new BufferedInputStream(inputStream)){
byte[] bytes=new byte[1024];
while (bufferedInputStream.read(bytes)>0){
System.out.println(new String(bytes, StandardCharsets.UTF_8));
}
}
复制代码
明白了ClassLoader#getResource()
,其实本篇文章就差很少了,由于后面要将的几个方法,底层都是ClassLoader#getResource()
class##getResource()
底层就是ClassLoader#getResource()
public java.net.URL getResource(String name) {
name = resolveName(name);
ClassLoader cl = getClassLoader0();
if (cl==null) {
// A system class.
return ClassLoader.getSystemResource(name);
}
return cl.getResource(name);
}
复制代码
不过有个小区别就在于class#getResource()
多了一个resolveName()
方法:
private String resolveName(String name) {
if (name == null) {
return name;
}
if (!name.startsWith("/")) {
Class<?> c = this;
while (c.isArray()) {
c = c.getComponentType();
}
String baseName = c.getName();
int index = baseName.lastIndexOf('.');
if (index != -1) {
name = baseName.substring(0, index).replace('.', '/')
+"/"+name;
}
} else {
name = name.substring(1);
}
return name;
}
复制代码
这个resolveName()
大体就是判断路径是相对路径仍是绝对路径,若是是相对路径,则资源名会被加上当前项目的根路径:
Test.class.getResource("spring-config.xml");
复制代码
resolve以后变成
com/dengchengchao/test/spring-config.xml
复制代码
这样的资源就只能在当前项目中找到。
Test.class.getResource("test.txt"); //相对路径
Test.class.getResource("/"); //根路径
复制代码
注意:
ClassLoader#getResource()
不能以/
开头
在Spring
中,对Resource
进行了扩展,使得Resource
可以适应更多的应用场景,
不过ClssPathResource()
底层依然是ClassLoader##getResource()
,所以ClassLoader##getResource()
d的特性,ClassPathResource
也支持。
protected URL resolveURL() {
if (this.clazz != null) {
return this.clazz.getResource(this.path);
} else {
return this.classLoader != null ? this.classLoader.getResource(this.path) : ClassLoader.getSystemResource(this.path);
}
}
复制代码
ClassPathResource
用于读取classes
目录文件
通常来讲,对于SpringBoot
项目,打包后的项目结构以下:
xxx.jar
|--- BOOT-INF
|--------|--classes
|--------|----|--com
|--------|----|-- application.properties
|--------|----|--logback.xml
| -------|-- lib
|--- META-INF
|--- org
能够看到,ClassPathResource()
的起始路径即是classes
,平时咱们读取的application.properties
即是使用ClasspathResource()
获取的
在平时使用的过程当中,有三点须要注意:
classpath 和 classpath* 区别:
classpath:只会返回第一个查找到的文件 classpath*:会返回全部查找到的文件
在Spring
中,须要直接表示使用ClassPathResource()
来查找的话,能够直接添加classpath:
头
使用classpath
以/
和不以/
开头没有区别
ServletContextResource
是针对Servlet
来作的,咱们知道,Servlet
规定webapp
目录以下:
而ServletContextResource
的路径则是xxx
目录下为起点。也就是能够经过ServletContextResource
获取到form.html
等资源。
同时对比上面的ClassPathResource
咱们能够发现:
"classpath:com"
复制代码
等价于:
ServletContextResource("WEB-INF/classes/com")
复制代码
FileSystemResource
没什么好说的,就是系统目录资源,好比
ApplicationContext ctx =
new FileSystemXmlApplicationContext("D://test.xml");
复制代码
它的标记头为file:
例如:
ApplicationContext ctx =
new FileSystemXmlApplicationContext("flie:D://test.xml");
复制代码
若是以为写得不错,欢迎关注微信公众号:逸游Java ,天天不定时发布一些有关Java进阶的文章,感谢关注