SpringBoot读取Resource下文件的几种方式读取jar里的excel,文件损坏

news/2024/7/21 6:06:41 标签: spring boot, jar, excel

在项目中涉及到Excle的导入功能,通常是我们定义完模板供用户下载,用户按照模板填写完后上传;
这里待下载模板位置为resource/template/员工基础信息导入模板.xlsx,
分别尝试了四种读取方式,并且测试了四种读取方式分别的windows开发环境下(IDE中)读取和生产环境(linux下springboot 的 jar包运行读取)。

文件后缀也正确 ,就是打不开
在这里插入图片描述

第一种:

excel__8"># 注意是 Springboot 下的 ClassPathResource , 而不是hutool 下的,否则报错,下载下来excel 后打不开
# 注意是 Springboot 下的 ClassPathResource , 而不是hutool 下的,否则报错,下载下来excel 后打不开 
ClassPathResource classPathResource = new ClassPathResource("template/员工基础信息导入模板.xlsx");
InputStream inputStream =classPathResource.getInputStream();

第二种:

InputStream inputStream = Thread.currentThread().getContextClassLoader().getResourceAsStream("template/员工基础信息导入模板.xlsx");

第三种:

InputStream inputStream = this.getClass().getResourceAsStream("template/员工基础信息导入模板.xlsx");

第四种:

File file = ResourceUtils.getFile("classpath:template/员工基础信息导入模板.xlsx");
InputStream inputStream = new FileInputStream(file);
经测试:

前三种方法在开发环境(IDE中)和生产环境(linux部署成jar包)都可以读取到;

第四种只有开发环境 时可以读取到,生产环境读取失败。

推测

主要原因是springboot内置tomcat,打包后是一个jar包,Resource下的文件是存在于jar这个文件里面,在磁盘上是没有真实路径存在的,它其实是位于jar内部的一个虚拟路径。所以通过ResourceUtils.getFile或者this.getClass().getResource(“”)方法无法正确获取文件。

只能通过类加载器读取。
前三种都可以读取到其实殊途同归,直接查看底层代码都是通过类加载器读取文件流,类加载器可以读取jar包中的编译后的class文件,当然也是可以读取jar包中的excle模板了。
用解压软件打开jar包查看结果如下:

其中cst文件中是编译后class文件存放位置,excleTemplate是模板存放位置,类加载器读取的是cst下class文件,同样可以读取excleTemplate下的模板的文件流了。

总结一下:

假如文件是在jar包中,读取方式应当使用基于类加载器读取文件流的方式,比如前三种方法;使用基于java中File方式的读取,在jar包情况下是读取不到的,比方说第四种。

    /**
     *  下载模板
     *
     * @param fileName          文件名称
     * @param response
     * @return
     * @author wangl
     * @date 2023-10-23
     */
    public static void downloadExcelTemplate(String fileName, HttpServletResponse response) {
        //方法一:直接下载路径下的文件模板(这种方式貌似在SpringCloud和Springboot中,打包成JAR包时,可能会无法读取到指定路径下面的文件)
        try {
            //设置要下载的文件的名称
            resetResponse(fileName, response);
            //通知客服文件的MIME类型
//            response.setContentType("application/vnd.ms-excel;charset=UTF-8");
            // 获取文件的路径
            // 注意是 Springboot 下的 ClassPathResource , 而不是hutool 下的,否则报错,下载下来excel 后打不开 
            ClassPathResource classPathResource = new ClassPathResource("/template/" + fileName);
            InputStream inputStream = classPathResource.getInputStream();
            OutputStream out = response.getOutputStream();
            byte[] b = new byte[2048];
            int len;
            while ((len = inputStream.read(b)) != -1) {
                out.write(b, 0, len);
            }
            //修正 Excel在“xxx.xlsx”中发现不可读取的内容。是否恢复此工作薄的内容?如果信任此工作簿的来源,请点击"是"
            response.setHeader("Content-Length", String.valueOf(inputStream.readAllBytes().length));
            inputStream.close();
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }



http://www.niftyadmin.cn/n/5116714.html

相关文章

嵌入式软件工程师——2025校招专题(一)

说明: 面试题来源于网络书籍,公司题目以及博主原创或修改(题目大部分来源于各种公司);文中很多题目,或许大家直接编译器写完,1分钟就出结果了。但在这里博主希望每一个题目,大家都要…

【USRP】通信总的分支有哪些

概述 通信是一个广泛的领域,涵盖了许多不同的技术、应用和专业分支。以下是通信领域的一些主要分支: 有线通信:这涉及到利用物理媒介(如电缆、光纤)进行通信。 电信:包括电话、电报和传真服务。宽带&#…

C算法:米m 转化 英尺foot和英寸inch

需求: 已知1英尺等于12英寸,如果已知英制长度的英尺foot和英寸inch的值,那么对应的米是(footinch/12)*0.3048,现在,用户输入的是厘米数,那么对应英制长度的英尺和英寸是多少呢? 输入样例: 1…

Dockerfile手册

大家好,我叫徐锦桐,个人博客地址为www.xujintong.com。平时记录一下学习计算机过程中获取的知识,还有日常折腾的经验,欢迎大家访问。 前言 一些自己经常用到的Dockerfile知识。Dockerfile就是构建自己的镜像的一个配置文件。 一、…

CentOS7安装部署CDH6.2.1

文章目录 CentOS7安装部署CDH6.2.1一、前言1.简介2.架构3.环境 二、环境准备1.部署服务器2.安装包准备3.修改机器名4.关闭防火墙5.关闭 SELinux6.Hosts文件7.limits文件8.设置swap空间9.关闭透明巨页内存10.免密登录 三、安装CM管理端1.安装第三方依赖包2.安装Oracle的JDK3.安装…

VsCode密钥链接远程服务器

大家好,我叫徐锦桐,个人博客地址为www.xujintong.com。平时记录一下学习计算机过程中获取的知识,还有日常折腾的经验,欢迎大家来访。 VsCode链接远程服务器的时候每次都要密码很麻烦吧,这是我们只要在本地生成了密钥&a…

TCP通信实战案例-模拟BS系统[了解]

前言 1、之前的客户端都是什么样的? 其实就是CS架构,客户端实需要我们自己开发实现的。 2、BS结构是什么样的,需要开发客户端吗? 浏览器访问服务端,不需要开发客户端。 实现BS开发 注意:服务器必须给浏…

轻松学习Python:基础知识汇总

轻松学习Python:基础知识汇总 目录 轻松学习Python:基础知识汇总 本文章向大家介绍轻松学习Python:基础知识汇总,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。 Python基础01 Hello World! Python命…