在开发过程当中咱们常常会碰到要在代码中获取资源文件的状况,而我在最近将原有的Tomcat的原生项目迁移到SpringBoot项目中时碰到一个问题,就是在本地运行时,获取本地的xml资源文件是可以获取到的,可是项目打成war包而后将其部署到Tomcat中运行时,就会发生问题,报找不到资源文件的错误。而后通过寻找排查肯定了是下面代码经过ClassLoader
获取路径的时候出错了。html
ExcelXmlModelFactory.class.getClassLoader().getResource("template/").getPath()
个人资源文件存放路径以下java
在本地中打印的日志路径为git
/Users/hupengfei/git/lap/lap-service/out/production/resources/template
可是在将SpringBoot打包成war包部署到Tomcat中时打印的目录为app
/home/app/lap/app/lap-service-1.0.0-SNAPSHOT.war!/WEB-INF/classes!/template/
能够看到在Linux中没法直接访问未经解压的文件,因此就会找不到文件。this
经过ClassLoader
的getResourceAsStream()
方法获取其流,就可以获取到.net
读取jar里面的文件,咱们只能用流去读取,不能用File3d
一般在开发过程当中会碰到读取配置文件的问题,通常有两种方式进行读取。一种是Class.getResource(String path)
,一种是ClassLoader.getResource(String path)
,这两种虽然都能读取文件,可是在path
的填写上有一点点的不一样。日志
/
开头:则是从ClassPath根下获取/
开头:默认是今后类所在的包下取资源下面有个例子code
public class Test { public static void main(String[] args) { System.out.println(Test.class.getResource("/")); System.out.println(Test.class.getResource("")); } }
输出以下xml
file:/Users/hupengfei/git/Test/out/production/classes/ file:/Users/hupengfei/git/Test/out/production/classes/Practice/Day13/
那么若是在resource
下有三个资源文件
那么该怎么获取这三个文件呢,由于在class文件夹中的目录结构以下
-- classes -- Convience -- Practice -- Test
因此若是想要获取Test下的资源文件,就以下的获取方法
System.out.println(Test.class.getResource("../../Test/1.xml")); System.out.println(Test.class.getResource("/Test/1.xml"));
ClassLoader.getResource
的path中不能以/
开头,path是默认是从根目录下进行读取的
例子以下
System.out.println(Test.class.getClassLoader().getResource("")); System.out.println(Test.class.getClassLoader().getResource("/"));
打印以下
file:/Users/hupengfei/git/Test/out/production/classes/ null
从上面例子咱们能够看到
Test.class.getClassLoader().getResource("")=Test.class.getResource("/")
其实查看Class.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); }
他最后调用的仍是ClassLoader.getResource
这个方法,那么为何会有path
的差异呢,由于其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; }
能够看到在这穿进去的为/
传出的是