来看看下面代码的结果java
package com.jdk.resource; public class Resource { public static void main(String[] args) throws Exception { System.out.println(Resource.class.getResource("")); System.out.println(Resource.class.getResource("/")); System.out.println(Resource.class.getClassLoader().getResource("")); System.out.println(Resource.class.getClassLoader().getResource("/")); } }
结果:api
file:/D:/workspace/JDKCore/bin/com/jdk/resource/ file:/D:/workspace/JDKCore/bin/ file:/D:/workspace/JDKCore/bin/ null
能够看到Resource.class.getResource("")获得的路径classpath下Resource 类所在包,而Resource.class.getResource("/")为classpath根路径;而Resource.class.getClassLoader().getResource("")一样为classpath根路径,Resource.class.getClassLoader().getResource("/")则为空。 下面来看看Class#getResource方法的源码this
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); }
从 cl.getResource(name)一行能够看出,Class#getResource方法最终仍是调用了CalssLoader#getResource方法。再来看看name = resolveName(name)的实现,作了什么处理,其调用了Class类的resolveName方法url
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方法的目的是当参数name以'/'开头就将'/'去除再返回,而当不以'/'开头则根据这个类对应的带包名全称变换成具体的路径名,如com.jdk.resource替换成com/jdk/resource。spa
如今应该明白最开始那个Resource 类中main方法的结果,实际上Class#getResource方法是去调CalssLoader#getResource的方法,只是在调用时会去判断是从相对路径仍是绝对路径获取资源。不过api是有一点点坑Class#getResource以'/'开头与ClassLoader#getResource相同,再记住ClassLoader#getResource不能以'/'开头就好了。.net
有了前面这个Classr#getResource与ClassLoader#getResource的对比,Class#getResourceAsStream与ClassLoader#getResourceAsStream其实也就ok了。ssr
Class#getResourceAsStream源码:code
public InputStream getResourceAsStream(String name) { name = resolveName(name); ClassLoader cl = getClassLoader0(); if (cl==null) { // A system class. return ClassLoader.getSystemResourceAsStream(name); } return cl.getResourceAsStream(name); }
ClassLoader#getResourceAsStream源码:接口
public InputStream getResourceAsStream(String name) { URL url = getResource(name); try { return url != null ? url.openStream() : null; } catch (IOException e) { return null; } }
因此,能够用下面三种方式获得资源ip
在com.jdk.resource目录下建立一个test.properties文件,能够用如下三种方式获得
package com.jdk.resource; public class Resource { public static void main(String[] args) throws Exception { //第一种方式 System.out.println(Resource.class.getResourceAsStream("test.properties")); //第二种方式 System.out.println(Resource.class.getResourceAsStream("/com/jdk/resource/test.properties")); //第三种方式 System.out.println(Resource.class.getClassLoader().getResourceAsStream("com/jdk/resource/test.properties")); } }
结果:
java.io.BufferedInputStream@659e0bfd java.io.BufferedInputStream@2a139a55 java.io.BufferedInputStream@15db9742
这对查看,Spring Resource 接口的实现ClassPathResource类getInputStream方法有必定帮助
public InputStream getInputStream() throws IOException { InputStream is; if (this.clazz != null) { is = this.clazz.getResourceAsStream(this.path); } else if (this.classLoader != null) { is = this.classLoader.getResourceAsStream(this.path); } else { is = ClassLoader.getSystemResourceAsStream(this.path); } if (is == null) { throw new FileNotFoundException(getDescription() + " cannot be opened because it does not exist"); } return is; }