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

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6

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

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

第一种

# 注意是 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();
        }
    }


阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6
标签: Spring