简单版:resources 目录下放模板 excel 文件,通过接口下载后,有文件,但打不开。

解决:maven 构建时对该 excel 模板进行了过滤,导致文件损坏,解决办法,在过滤的时候把 xlsx 排除掉(<nonFilteredFileExtension>xlsx</nonFilteredFileExtension>)。

<build>
	<plugins>
    	<plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-resources-plugin</artifactId>
                <configuration>
                    <nonFilteredFileExtensions>
						<!--不加这一行,xlsx文件会被过滤,然后在maven build的时候,去target下看对应的xlsx就是损坏的-->
                        <nonFilteredFileExtension>xlsx</nonFilteredFileExtension>
                    </nonFilteredFileExtensions>
                </configuration>
        </plugin>
    </plugins>
</build>

具体过程

问题: springboot 项目简单的下载excel 模板功能,模板放在resources/template/目录中,

public static void downloadTemplate(String downloadName, String classPath) {
    // 处理下中文乱码问题获取response
    HttpServletResponse response = ResponseUtils.getHttpResponse(downloadName + XLSX);
    try (InputStream inputStream = new ClassPathResource(classPath).getStream();
         OutputStream outputStream = response.getOutputStream()) {
        IOUtils.copy(inputStream, outputStream);
    } catch (IOException e) {
        log.error(String.format("下载%s失败", downloadName), e);
    }
}

代码挺简单,一运行,也挺顺利,很快就把文件下好了。点开看看,咦,怎么回事,怎么提示我可能是内存不足,不足以打开这个文件呢?

尝试一

当时电脑也有点卡,心想可能是内存不够了,关了一些文件,还是无法正常打开;而打开那个模板文件就是正常的。排除提示的情况, 确定是文件出错了。

尝试二

代码是直接从之前做的项目直接拷贝的,之前运行的都没问题,按理来说,也应该没问题啊。直接去搜 java excel下载无法打开, 有说什么 Response 要设置 Content-Length 才行。试了一下还是不行。

尝试三

直接尝试从本地文件的 excel 文件,写到另一个文件,发现可以正常写入;从 classpath读取,写入失败。也就是,从 classpath 读取的时候,就是一个有问题的 excel 文件。

尝试四

之前一直着眼于代码结构中的模板文件了,后来想到, 太傻了, 要去 target 目录下去看看才是。打开一看,好家伙,编译过后的 excel 文件就是损坏的。

尝试五

这个时候才想到搜索时有文章提到 maven 的 filtering,发现是 <filtering>true</filtering>

<resources>
    <resource>
        <directory>src/main/resources</directory>
        <filtering>true</filtering>
    </resource>
    <resource>
        <directory>src/main/java</directory>
        <includes>
            <include>**/*.xml</include>
            <include>**/*.json</include>
            <include>**/*.ftl</include>
        </includes>
    </resource>
</resources>

当然,粗暴一点解决就是改为 false, 以前的项目也一直是 false;当中也试了很多 <excludes>,现在看起来是白费力气, 它喵的, <excludes> 就直接排除了,你重新 mvn clean compile 的时候,直接看不到 excel 模板了。最后还是采用了开头描述的方案。
至于为什么会损坏,只在官网文档(https://maven.apache.org/plugins/maven-resources-plugin/examples/filter.html)找到这样一句话:Warning: Do not filter files with binary content like images! This will most likely result in corrupt output.

总结

有的时候,程序报的错误会误导我们,尽管它告诉我们错误是由于“原因a”引起的,但排查到最后,可能是“原因z”引起的。在遇到问题的时候,我们就应该仔细思考一下,哪些是没问题的,有哪些可能导致了问题的发生。而不是被某个错误原因所误导。

虽然问题还是比较简单,但还是耗费了不少时间趟这条弯路。因此记录一下,作为警示。

当你准备好了,机会来临的时候,你才能抓住