xlsx文件下载异常问题

前端调用我的后端接口下载xlsx文件,里面的数据没有正常展示。如下所示:

查看前端响应头里面。多了个charset=utf-8;

java 复制代码
application/octet-stream;charset=utf-8

如果是下面这种响应不会有问题,但是上面二进制流有charset=utf-8,浏览器会把这个当文本处理,不会解析成文件。

改动之前下载接口如下:

java 复制代码
  @ApiOperation(value = "oss文件下载")
    @GetMapping("downOssFile")
    public ResponseEntity<byte[]> getFileByUrl(String url, String fileName) {
        byte[] body = ossManager.download(url);
        HttpHeaders headers = new HttpHeaders();

        headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);
        try {
            String encodeFileName = URLEncoder.encode(fileName, "utf-8").replaceAll("\\+", "%20");
            headers.set("Content-Disposition", "attachment;filename=" + encodeFileName);
        } catch (UnsupportedEncodingException e) {
            log.error("oss文件下载失败,e:", e);
            throw new RuntimeException(e);
        }
        ResponseEntity<byte[]> result = ResponseEntity.ok().headers(headers).body(body);
        return result;
    }

这个接口返回体是使用Spring的ResponseEntity<byte\[\]> , Spring 的 ByteArrayHttpMessageConverter 会序列化处理该响应体。 最后过滤器在 response 提交后仍可修改 header。

检查我的web.xml文件配置是这样的,强制在所有接口的请求的响应头里面加上charset=UTF-8设置。这是问题的根源所在。

java 复制代码
    <filter>
		<filter-name>ProjectEncoding</filter-name>
		<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
		<async-supported>true</async-supported>
		<init-param>
			<param-name>encoding</param-name>
			<param-value>UTF-8</param-value>
		</init-param>
		<init-param>
			<param-name>forceEncoding</param-name>
			<param-value>true</param-value>
		</init-param>
	</filter>
	<filter-mapping>
		<filter-name>ProjectEncoding</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>

下面是我修改之后的下载接口:

java 复制代码
@ApiOperation(value = "oss文件下载")
    @GetMapping("downOssFile")
    public void getFileByUrl(String url, String fileName, HttpServletResponse response) throws IOException {
        byte[] body = ossManager.download(url);
        response.setHeader("Content-Type", "application/octet-stream");
        String encodeFileName = URLEncoder.encode(fileName, "utf-8").replaceAll("\\+", "%20");
        response.setHeader("Content-Disposition", "attachment;filename=" + encodeFileName);
        response.getOutputStream().write(body);
    }

使用原生的Response方法,绕过过滤器,getOutputStream().write() 会提前提交 response,过滤器无法再改charset。这样xlsx文件才能正常解析。

这个问题确实也排查了好久,总结来说对于上传以及下载文件异常问题首先考虑的就是响应头或者请求头,一般问题都出在这两个。

相关推荐
小bo波1 小时前
使用Thread子类创建线程 VS 使用Runnable接口创建线程的区别
java·多线程·thread·并发编程·runnable
SamDeepThinking2 小时前
高并发场景下,CompletableFuture与ForkJoinPool该如何取舍?
java·后端·面试
张不才5 小时前
CPU 100% 了怎么办?Java 性能排障的标准化操作
java·后端
shepherd1116 小时前
吞吐量提升 10 倍:高并发大批量数据处理任务的架构演进与性能调优
java·后端·架构
plainGeekDev9 小时前
单例模式 → object 声明
android·java·kotlin
用户2986985301410 小时前
Java 实现 Word 文档文本与图片提取的方法
java·后端
SimonKing11 小时前
铁子,IntelliJ IDEA 2026.1.3来了,升不升?
java·后端·程序员
咖啡八杯1 天前
GoF设计模式——策略模式
java·后端·spring·设计模式
用户128526116021 天前
我把祖传Java项目重构后,接口响应从3s砍到了200ms,只改了这几行代码
java