Spring Boot 无法加载 ClasspathResource 问题

本文撰写时的 Spring 版本: Spring Boot 2.0.5.RELEASE

一般来说, 程序运行所必须的资源文件我们会一起打包到 jar. 那么接下去我们就要读取这个资源文件.

假定我们的资源文件在源码目录中为 src/main/resources/myFile.txt

Spring Boot 中读取存放在 classpath 的资源文件通常是这么做的

val myFile = ClasspathResource("/myFile.txt")

(不同于 Class.getResourceAsStream(name:String), ClasspathResource 的参数可以去除第一个 /)

然后我们读取这个文件的内容通常是先获取他的流

myFile.inputStream

这好像没有什么问题, 文件也正常的被读取了.

一些第三方库可能要求传入一个 File 类型.

myFile.file

看起来也没有问题, 并且在 IDEA 里运行的时候确实没有问题. 直到 CI 测试的时候, 就会有这么一个异常

java.io.FileNotFoundException: class path resource [myFile.txt] cannot be resolved to absolute file path because it does not reside in the file system: jar:file:/app.jar!/BOOT-INF/classes!/myFile.txt

在集成环境和生产环境上, 我们的程序是一个 jar 包而不是 exploded 方式, 也就是说, 此时的 Resource.getFile() 将有不一样的行为.

我们在 IDEA 调试的时候, 资源文件是存在于真实文件系统里的一个文件. 而在 jar 包中, 它不是一个真实文件系统的文件.

为了能用统一的文件系统路径去表示 jar 内的文件, Java 开创了 ! 这个符号.

! 表示这个文件是一个压缩包(zip)(jar 本身就是一个 zip), 之后的路径则为压缩包内的路径(压缩包内的路径不分运行平台, 统一为 Unix 路径).

正常情况下的 getFile() 操作, 会得到一个 jar 包路径后面加上一个 ! 号然后再拼接上包内路径的一个路径.

Spring Boot 为了避免资源文件冲突(Java 的打包规范忽略了资源文件的问题, 两个库的代码文件是可以合并的, 因为包名不同. 但是资源文件都从 jar 的根目录开始编排, 如果重名将互相覆盖而导致打包后资源文件的丢失)而采用 fat-jar 的方式来打包程序.

fat-jar 就是一种 nested jar, 所有的依赖库不会合并到用户代码上, 而是以 jar 包的形式存放在 jar 包内.

一个典型的 Spring Boot 程序打包后差不多是这样的

META-INF
    MANIFEST.MF
org
    springframework
        boot
            loader
                (Launcher)
BOOT-INF
    classes
        (user code)
    lib
        dependence.jar

jar 的入口类其实是 Spring Boot Launcher, 他会为每一个依赖创建一个 ClassLoader, 这样就可以让每个依赖自己读取自己的资源文件而互不冲突.

而用户自己的类是从 /BOOT-INF/classes 开始的, 用户自己的资源文件的根目录也在这里, 所以为了让用户能够正确读到自己的资源文件. 加载用户代码的那个 ClassLoaderclasspath 从这里开始.

fat-jar 并不是 Java 官方标准, 所以 Java 认为所有 classpath 都是从 jar 的根目录开始的.

于是我们得到的文件路径, 将是 {用户代码根目录}!/{资源文件路径}

而用户代码根目录本身就是在 jar 内的, 最终我们会得到这么一个路径

jar:file:/app.jar!/BOOT-INF/classes!/myFile.txt

(注意, 有两个 ! 号)

没错, classes 文件夹被认为是一个压缩包了.

所以我们将找不到这个文件.

如果读取资源文件的操作只在自己的代码发生, 那么只要不使用 Resource.getFile() 而直接获取流就可以避免这个问题. 但是很多情况下, 并非自己要读文件, 而是第三方库要读文件.

例如第三方库可能会要求在配置文件中配置 key 文件的路径, 而这个路径支持网络读取, 所以必须是 URI. 然后第三方库的代码中就会使用 Resource.getFile() 来把这个地址转成 File 类型再去读他.

这么一读, 就抛出异常了.

那么, 怎么办呢.

我们找到了这么一个库 https://github.com/ulisesbocchio/spring-boot-jar-resources

他的功能是通过自定义的 ResourceLoader, 当 Spring Boot 需要读取文件时, 首先判断这个文件是不是存在于 classpath 中, 如果是, 则解压这个文件到临时目录(真实文件系统上), 然后返回文件系统路径.

使用非常简单, 首先加入依赖

// https://mvnrepository.com/artifact/com.github.ulisesbocchio/spring-boot-jar-resources
compile group: 'com.github.ulisesbocchio', name: 'spring-boot-jar-resources', version: '1.3'

然后把程序入口改成这样就行了

@SpringBootApplication
open class Application

fun main() {
    runApplication<Application> {
        resourceLoader = JarResourceLoader()
    }
}

我们再使用 ApplicationContext.getResource() 时, 返回的就不是 ClasspathResource 了, 而是 JarResource, 路径在一个临时目录下(Linux 下默认为 /tmp/**)

这样, 我们就可以让第三方库正常工作了.