一般的web系统基本都会有文件上传功能,文件上传必然涉及到一个问题,就是文件大小,太大的文件不仅传输速度慢,而且对服务器压力巨大,后期的下载和保存都是一种考验。
所以有了文件大小限制,一般我们在处理上传的接口时,其实就可以判断文件大小,这时候做一个大小判断,不符合要求,直接不保存文件,并返回前端相关提示。这种做法有一个不好的地方,就是如果上传方法重载,或者使用的地方不一样,那么每个方法都需要做这样的大小限制判断。
springboot使用MultipartFile上传,可以对文件上传大小做限制。直接配置如下所示:
spring.servlet.multipart.enabled=true
spring.servlet.multipart.max-file-size=20MB
spring.servlet.multipart.max-request-size=200MB
这种设置,如果不做额外的处理,的确可以起到作用,但是如果你调用上传接口,上传过大文件,返回的结果可能如下所示:
{
"timestamp": "2022-04-09T06:39:28.820+00:00",
"status": 500,
"error": "Internal Server Error",
"path": "/test/upload"
}
这个是springboot通过拦截器做出的返回,而且后台还会打印异常信息。
有时候,我们希望我们的返回信息跟我们其他的业务逻辑返回一样,就需要自定义一个返回。
做法很简单,自定义一个BaseController,然后让上传的Controller来继承这个类即可。
package com.example.undertowtest.web;
import io.undertow.server.handlers.form.MultiPartParserDefinition;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.multipart.MultipartException;
import java.util.HashMap;
import java.util.Map;
@ControllerAdvice
public class BaseController {
@ResponseBody
@ExceptionHandler(value = MultipartException.class)
public Object fileUploadExceptionHandler(MultipartException exception){
Map<String,Object> map = new HashMap<>();
map.put("code",500);
Throwable rootCause = exception.getRootCause();
if(rootCause instanceof MultiPartParserDefinition.FileTooLargeException){
map.put("msg","文件太大");
}
return map;
}
}
这个时候,如果上传过大的文件,那么返回信息如下所示:
{
"msg": "文件太大",
"code": 500
}
这种code,msg可以根据自己的业务定制化,最终前端在处理的时候就可以和其他业务一样处理,而不是去判断status,error等模糊的返回信息。这样,后端也不会打印文件上传的异常堆栈信息。