此文要从SpringBoot打包后不能读取classpath下文件说起

简介: 您好,我是码农飞哥,感谢您阅读本文!如果此文对您有所帮助,请毫不犹豫的给个一键三连吧。

您好,我是码农飞哥,感谢您阅读本文!如果此文对您有所帮助,请毫不犹豫的给个一键三连吧。

此文要从SpringBoot打成jar之后不能读取classpath下文件说起,并由此作为一个切入点,思考如何正确的读取jar包中的文件。

问题复现

事情是这样的,昨天快下班了时候,测试小姐姐突然说(PS: 有些测试小姐姐上班的时候不提啥BUG,下班的时候给你提一堆BUG,不知道大家有没有这种感觉)。

有个图片下载的接口不能用了。害,本想6点钟下班走人的我瞬间懵逼了,这下走不了。老规矩,先查错误日志。

还是那个熟悉的FileNotFoundException异常。

2021-04-21 09:43:59.715  INFO 16896 --- [nio-8383-exec-1] com.jay.ImgDownloadController            : qr_code-icon path is: file:/D:/workspace/file_dow
nload_demo/target/file_download_demo-0.0.1-SNAPSHOT.jar!/BOOT-INF/classes!/img/qr_code-icon.png
java.io.FileNotFoundException: file:\D:\workspace\file_download_demo\target\file_download_demo-0.0.1-SNAPSHOT.jar!\BOOT-INF\classes!\img\qr_code-icon.
png (文件名、目录名或卷标语法不正确。)
        at java.io.FileInputStream.open0(Native Method) ~[na:1.8.0_60]
        at java.io.FileInputStream.open(Unknown Source) ~[na:1.8.0_60]
        at java.io.FileInputStream.<init>(Unknown Source) ~[na:1.8.0_60]
  at com.jay.ImgDownloadController.downloadImage(ImgDownloadController.java:35) ~[classes!/:0.0.1-SNAPSHOT]

很明显这是一个无效的文件路径,根据这个错误路径程序肯定不能找到指定的文件咯。

再回头定位到报错的代码

//读取文件的路径
        String path = Thread.currentThread().getContextClassLoader().getResource("").getPath()
                + "img/qr_code-icon.png";
        LOGGER.info("qr_code-icon path is: " + path);
  InputStream is = new FileInputStream(new File(path));

这段代码在本地调试的时候明明是没有问题的呀,单元测试都跑过了,本地调试时输入的路径如下:。

13df44613c3417aa7764d478cf16240c_20210421103609496.png

打印的路径地址是:D:/workspace/file_download_demo/target/classes/img/qr_code-icon.png

这个路径是一个有效的路径。

这又是一个我本地明明没问题,到服务器就有问题了。锅是甩不出去了

问题思考

我们都知道JAVA是一门静态语言,先编译再运行也就是先将java文件编译成class文件,然后在用虚拟机来执行class文件的。SpringBoot在编译打包后会生成target目录,class文件,资源文件还有jar包都会被放在这个目录下。如下图所示:

0042bcdbd13e80a4c2c75d36ed2d9298_20210421105019635.png

其中所有的class文件以及资源文件都放在了classes文件夹中。在本地运行时 Thread.currentThread().getContextClassLoader().getResource("").getPath()

其中Thread.currentThread().getContextClassLoader()返回的是当前线程的类加载器(默认是AppClassLoader类加载器),类加载器可以加载类也可以加载资源。类加载器有很多,具体可以参考双亲委派模型以及SpringFactoriesLoader详解(最全最简单的介绍)

读取到的路径是D:/workspace/file_download_demo/target/classes,classes文件夹所在的路径也就是我们熟悉的classpath 路径 。

而通过jar包来运行时,上面的代码读取的是jar的绝对路径,而jar是一个压缩包,直接读取其包内的绝对路径是有问题的。 也就是会报上面的错误。

问题解决

既然不能通过路径的方式来获取jar中文件,那么该通过何种方式来获取呢?这里有两种写法。

1.通过ClassPathResource获取输入流的方式

InputStream is = new ClassPathResource("img/qr_code-icon.png").getInputStream();

2.通过getResourceAsStream方法获取输入流

InputStream is = Thread.currentThread().getContextClassLoader()
                .getResourceAsStream("img/qr_code-icon.png");

可以看出上面两种都是直接获取文件流的方式获取文件,那么问题来了,为啥这种方式可以呢?因为在jar文件中不能直接通过资源路径的方式获取文件,但是可以在jar包中拿到文件流。

测试结果

总结

本文从SpringBoot打成jar之后不能读取classpath下文件说起,介绍了为啥打成jar之后不能通过路径的方式访问classpath下的文件,接着说明了如何处理这个问题,最后介绍了通过流的方式来处理这个问题。

结尾彩蛋

1.Thread.currentThread().getContextClassLoader().getResource("").getPath() 这种写法在通过War运行的项目(比如一个Sping MVC项目)中获取classpath下的文件有没有问题呢?欢迎知道的小伙伴积极留言。

2.通过ResouceUtils.getFile()的方式能不能获取到classpath下文件呢?

源码

@RestController
public class ImgDownloadController {
    private static final Logger LOGGER = LoggerFactory.getLogger(ImgDownloadController.class);
    /**
     * 图片下载接口
     *
     * @param response
     * @Author xiang.wei
     */
    @RequestMapping("/download/image")
    public void errorDownloadImage(HttpServletResponse response) throws IOException {
        //读取文件的路径
        String path = Thread.currentThread().getContextClassLoader().getResource("").getPath()
                + "img/qr_code-icon.png";
        LOGGER.info("qr_code-icon path is: " + path);
        InputStream is = new FileInputStream(new File(path));
        downloadFile(is, "qr_code-icon.png", response);
    }
    @RequestMapping("/correct/download/image1")
    public void correctDownloadImage1(HttpServletResponse response) throws IOException {
        InputStream is = new ClassPathResource("img/qr_code-icon.png").getInputStream();
        downloadFile(is, "qr_code-icon.png", response);
    }
    @RequestMapping("/correct/download/image2")
    public void correctDownloadImage2(HttpServletResponse response) throws IOException {
        InputStream is = Thread.currentThread().getContextClassLoader()
                .getResourceAsStream("img/qr_code-icon.png");
        downloadFile(is, "qr_code-icon.png", response);
    }
    private void downloadFile(InputStream is, String name, HttpServletResponse response) throws IOException {
        //设置响应头  "application/octet-stream"
        response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE);
        response.setHeader("Content-Disposition", "attachment;filename=" + java.net.URLDecoder.decode(name, "ISO-8859-1"));
        //输出流自动关闭
        try (OutputStream os = response.getOutputStream()) {
            byte[] buffer = new byte[1024];
            int len = 0;
            while ((len = is.read(buffer)) != -1) {
                os.write(buffer, 0, len);
                os.flush();
            }
        } finally {
            if (is != null) {
                is.close();
            }
        }
    }
}


相关文章
|
3月前
|
前端开发 JavaScript Java
SpringBoot项目部署打包好的React、Vue项目刷新报错404
本文讨论了在SpringBoot项目中部署React或Vue打包好的前端项目时,刷新页面导致404错误的问题,并提供了两种解决方案:一是在SpringBoot启动类中配置错误页面重定向到index.html,二是将前端路由改为hash模式以避免刷新问题。
330 1
|
1月前
|
Java 应用服务中间件
SpringBoot获取项目文件的绝对路径和相对路径
SpringBoot获取项目文件的绝对路径和相对路径
107 1
SpringBoot获取项目文件的绝对路径和相对路径
|
1月前
|
网络协议 Java
springboot配置hosts文件
springboot配置hosts文件
52 11
|
2月前
|
XML Java Kotlin
springboot + minio + kkfile实现文件预览
本文介绍了如何在容器中安装和启动kkfileviewer,并通过Spring Boot集成MinIO实现文件上传与预览功能。首先,通过下载kkfileviewer源码并构建Docker镜像来部署文件预览服务。接着,在Spring Boot项目中添加MinIO依赖,配置MinIO客户端,并实现文件上传与获取预览链接的接口。最后,通过测试验证文件上传和预览功能的正确性。
129 4
springboot + minio + kkfile实现文件预览
|
1月前
|
存储 前端开发 JavaScript
|
1月前
|
Dubbo Java 应用服务中间件
深入探讨了“dubbo+nacos+springboot3的native打包成功后运行出现异常”的原因及解决方案
本文深入探讨了“dubbo+nacos+springboot3的native打包成功后运行出现异常”的原因及解决方案。通过检查GraalVM版本兼容性、配置反射列表、使用代理类、检查配置文件、禁用不支持的功能、查看日志文件、使用GraalVM诊断工具和调整GraalVM配置等步骤,帮助开发者快速定位并解决问题,确保服务的正常运行。
54 1
|
1月前
|
存储 Java API
|
2月前
|
Java Docker 微服务
SpringBoot微服务打包Docker镜像
SpringBoot微服务打包Docker镜像
85 11
|
2月前
|
easyexcel Java UED
SpringBoot中大量数据导出方案:使用EasyExcel并行导出多个excel文件并压缩zip后下载
在SpringBoot环境中,为了优化大量数据的Excel导出体验,可采用异步方式处理。具体做法是将数据拆分后利用`CompletableFuture`与`ThreadPoolTaskExecutor`并行导出,并使用EasyExcel生成多个Excel文件,最终将其压缩成ZIP文件供下载。此方案提升了导出效率,改善了用户体验。代码示例展示了如何实现这一过程,包括多线程处理、模板导出及资源清理等关键步骤。
|
2月前
|
消息中间件 Java 大数据
大数据-56 Kafka SpringBoot与Kafka 基础简单配置和使用 Java代码 POM文件
大数据-56 Kafka SpringBoot与Kafka 基础简单配置和使用 Java代码 POM文件
79 2