SpringBoot读取war包jar包Resource资源文件解决办法

场景描述

在开发过程中我们经常会碰到要在代码中获取资源文件的情况,而我在最近在SpringBoot项目中时碰到一个问题,就是在本地运行时,获取本地的xml资源文件是能够获取到的,但是项目打成war包jar包启动运行时,就会发生问题,报找不到资源文件的错误。然后经过寻找排查确定了是下面代码通过ClassLoader获取路径的时候出错了。
 
常用方式:

/**
* @author mazhq
* @Title: TestMain
* @ProjectName: zeus
* @Description: TODO
* @date 2019/3/5 16:10
*/
public class TestMain {
public static void main(String[] args) {
String path = TestMain.class.getClassLoader().getResource("1.xml").getPath();
System.out.println(path);
}
 /**
* 输出:
*
*/D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml
*/
}

  

但是在将SpringBoot打包放到Linux服务器启动打印的目录为

/data/zeus/service-hi-1.0.0-SNAPSHOT.war!/WEB-INF/classes!/1.xml

可以看到在Linux中无法直接访问未经解压的文件,所以就会找不到文件。

解决办法

  1. 通过ClassLoadergetResourceAsStream()方法获取其流,就能够获取到。

  读取jar里面的文件,我们只能用流去读取,不能用File

public class TestMain {
public static void main(String[] args) {
try {
List<String> content = IOUtils.readLines(TestMain.class.getClassLoader().getResourceAsStream("1.xml"), "UTF-8");
} catch (IOException e) {
e.printStackTrace();
}
}
}

  2. 采用绝对路径将文件放到服务器某个路径,在application.properties中配置路径读取。

  3. 不推荐:将内容放到数据库中。

获取资源的两种方式

通常在开发过程中会碰到读取配置文件的问题,一般有两种方式进行读取。一种是Class.getResource(String path),一种是ClassLoader.getResource(String path),这两种虽然都能读取文件,但是在path的填写上有一点点的不同。

Class.getResource

  1. path以/开头:则是从ClassPath根下获取
  2. path不以/开头:默认是从此类所在的包下取资源

下面有个例子

public class TestMain {
public static void main(String[] args) {
System.out.println(TestMain.class.getResource("/"));
System.out.println(TestMain.class.getResource(""));
}
/**
* 输出:
*
* file:/D:/demo_projects/sc-architecture/service-hi/target/classes/
* file:/D:/demo_projects/sc-architecture/service-hi/target/classes/com/mazhq/servicehi/
*/
}

  

那么读取在resource下的1.xml,就如下的获取方法

public class TestMain {
public static void main(String[] args) {
System.out.println(TestMain.class.getResource("/1.xml"));
System.out.println(TestMain.class.getResource("../../../1.xml"));
}
/**
* 输出:
*
* file:/D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml
* file:/D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml
*/
}

ClassLoader.getResource

ClassLoader.getResource的path中不能以/开头,path是默认是从根目录下进行读取的

代码如下:

public class TestMain {
public static void main(String[] args) {
System.out.println(TestMain.class.getClassLoader().getResource(""));
System.out.println(TestMain.class.getClassLoader().getResource("/"));
}
/**
* 输出:
*
* file:/D:/demo_projects/sc-architecture/service-hi/target/classes/
* null
*/
}

  

从上面例子我们可以看到

TestMain.class.getClassLoader().getResource("")=TestMain.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方法中对传的/进行了解析,解析为了空字符串。

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;
}   //传入 "/" 返回  ""

  

最后:大家用的时候注意一下这些问题,避免在这个上面耽误时间。

最新文章

  1. ACM: NBUT 1105 多连块拼图 - 水题 - 模拟
  2. svg转化成canvas以便生成base64位的图片
  3. CSSOM视图模式(CSSOM View Module)相关整理:scrollWidth,scrollLeft,offsetLeft,clientX ,offsetX 定义和区别
  4. php 下载文件的函数
  5. BZOJ3640 : JC的小苹果
  6. wcf调用oracle存储过程
  7. LintCode-Fast Power
  8. 【转】Linux 技巧: Bash 参数和参数扩展
  9. 动态上传多个文件(asp)
  10. Microsoft Word Regular Expression
  11. maven 不同环境加载不同的properties 文件
  12. Eric5 for Python 3.3.3安装指南
  13. 使用Spring Session实现Spring Boot水平扩展
  14. poj-1904(强连通缩点)
  15. java Ajax跨域请求COOKIE无法带上的解决办法
  16. java线程入门一
  17. nginx安装与挂载
  18. K8S flannel
  19. net core体系-web应用程序-4asp.net core2.0 项目实战(1)-7项目缓冲方案( Redis)
  20. UVa-156 Ananagrams 反片语【map】【vector】

热门文章

  1. IO(字节流)
  2. linux下删除大量文件提示参数过长解决办法
  3. 负载均衡之-LVS
  4. mysql jdbc性能优化之mybatis/callablestatement调用存储过程mysql jdbc产生不必要的元数据查询(已解决,cpu负载减少20%)
  5. 20145330 《网络对抗》PC平台逆向破解:注入shellcode 和 Return-to-libc 攻击实验
  6. ZooKeeper参数调优
  7. DSDS,双模,双卡,双待,单待,双通,单通,概念及相互关系?【转】
  8. UVA 10462 Is There A Second Way Left?(次小生成树&amp;Prim&amp;Kruskal)题解
  9. POJ3241 Object Clustering(最小生成树)题解
  10. Unity3D学习笔记(十九):UGUI、Image、Text、Button