SpringBoot 上传文件判空以及格式检验流程
作者:?吾?非?水 发布时间:2023-01-19 05:07:36
标签:上传文件,判空,格式检验
基于jsr303 通过自定义注解实现,实现思路:
存在一些瑕疵,后续补充完善。
加入依赖
部分版本已不默认自动引入该依赖,选择手动引入
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-validation</artifactId>
</dependency>
创建自定义注解以及实现类
目录结构:
FileNotEmpty
自定义注解FileNotEmptyValidator
单文件校验FilesNotEmptyValidator
多文件校验
/**
* jsr303 文件格式校验注解
*
* @author maofs
* @version 1.0
* @date 2021 -11-29 10:16:03
*/
@Documented
@Constraint(
validatedBy = {FileNotEmptyValidator.class, FilesNotEmptyValidator.class}
)
@Target({ElementType.METHOD, ElementType.FIELD, ElementType.ANNOTATION_TYPE, ElementType.CONSTRUCTOR, ElementType.PARAMETER, ElementType.TYPE_USE})
@Retention(RetentionPolicy.RUNTIME)
public @interface FileNotEmpty {
/**
* Message string.
*
* @return the string
*/
String message() default "文件格式不正确";
/**
* 校验组
*
* @return the class [ ]
*/
Class<?>[] groups() default {};
/**
* Payload class [ ].
*
* @return the class [ ]
*/
Class<? extends Payload>[] payload() default {};
/**
* 需要校验的格式数组
*
* @return the string [ ]
*/
String[] format() default {};
/**
* 是否必填 为false时文件为空则不校验格式,不为空则校验格式
* 为true时文件不能为空且需要验证格式
*
* @return the boolean
*/
boolean required() default true;
/**
* 单文件校验
*
* @author maofs
* @version 1.0
* @date 2021 -11-29 10:16:03
*/
public class FileNotEmptyValidator implements ConstraintValidator<FileNotEmpty, MultipartFile> {
private Set<String> formatSet = new HashSet<>();
private boolean required;
@Override
public void initialize(FileNotEmpty constraintAnnotation) {
String[] format = constraintAnnotation.format();
this.formatSet = new HashSet<>(Arrays.asList(format));
this.required = constraintAnnotation.required();
}
@Override
public boolean isValid(MultipartFile multipartFile, ConstraintValidatorContext constraintValidatorContext) {
if (multipartFile == null || multipartFile.isEmpty()) {
return !required;
}
String originalFilename = multipartFile.getOriginalFilename();
assert originalFilename != null;
String type = originalFilename.substring(originalFilename.lastIndexOf('.') + 1).toLowerCase();
if (!formatSet.isEmpty()) {
return formatSet.contains(type);
}
return true;
}
}
/**
* 多文件校验
*
* @author maofs
* @version 1.0
* @date 2021 -11-29 10:16:03
*/
public class FilesNotEmptyValidator implements ConstraintValidator<FileNotEmpty, MultipartFile[]> {
private Set<String> formatSet = new HashSet<>();
private boolean required;
@Override
public void initialize(FileNotEmpty constraintAnnotation) {
String[] format = constraintAnnotation.format();
this.formatSet = new HashSet<>(Arrays.asList(format));
this.required = constraintAnnotation.required();
}
@Override
public boolean isValid(MultipartFile[] multipartFiles, ConstraintValidatorContext constraintValidatorContext) {
if (multipartFiles == null || multipartFiles.length == 0) {
return !required;
}
for (MultipartFile file : multipartFiles) {
String originalFilename = file.getOriginalFilename();
assert originalFilename != null;
String type = originalFilename.substring(originalFilename.lastIndexOf('.') + 1).toLowerCase();
if (formatSet.isEmpty() || !formatSet.contains(type)) {
return false;
}
}
return true;
}
}
全局异常处理
/**
* 统一异常处理
*
* @author maofs
* @version 1.0
* @date 2021 -11-29 10:16:03
*/
@ControllerAdvice
public class ExceptionHandle {
private final static Logger logger = LoggerFactory.getLogger(ExceptionHandle.class);
@ExceptionHandler(value = Exception.class)
@ResponseBody
public Result<String> handle(Exception e) {
logger.error(e.getMessage());
StringBuilder stringBuilder = new StringBuilder();
//jsr303异常
if (e instanceof ConstraintViolationException) {
ConstraintViolationException ex = (ConstraintViolationException)e;
Set<ConstraintViolation<?>> constraintViolations = ex.getConstraintViolations();
for (ConstraintViolation<?> constraintViolation : constraintViolations) {
stringBuilder.append(constraintViolation.getMessageTemplate());
}
} else if (e instanceof BindException) {
BindException bindException = (BindException)e;
stringBuilder.append(bindException.getFieldErrors()
.stream()
.map(FieldError::getDefaultMessage)
.collect(Collectors.joining(",")));
} else {
stringBuilder.append("未知错误:").append("请联系后台运维人员检查处理!");
}
return ResultUtil.fail(stringBuilder.toString());
}
}
使用示例
/**
* 文件上传示例接口
*
* @author maofs
* @version 1.0
* @date 2021 -11-19 16:08:26
*/
@RestController
@Validated
@RequestMapping("/annex")
public class AnnexController {
@Resource
private IAnnexService annexService;
/**
* 文件上传示例1
*
* @param uploadDTO the upload dto
* @return the result
*/
@PostMapping(value = "/upload1")
public Result<String> upload(@Valid AnnexUploadDTO uploadDTO) {
return Boolean.TRUE.equals(annexService.upload(uploadDTO)) ? ResultUtil.success() : ResultUtil.fail();
}
/**
* 文件上传示例2
*
* @param number 项目编号
* @param pictureFile 图片文件
* @param annexFile 附件文件
* @return result result
*/
@PostMapping(value = "/upload2")
public Result<String> upload(@NotBlank(@FileNotEmpty(format = {"png", "jpg"}, message = "图片为png/jpg格式", required = false)
MultipartFile pictureFile, @FileNotEmpty(format = {"doc", "docx", "xls", "xlsx"}, message = "附件为doc/docx/xls/xlsx格式", required = false)
MultipartFile annexFile) {
return Boolean.TRUE.equals(annexService.upload( pictureFile, annexFile)) ? ResultUtil.success() : ResultUtil.fail();
}
@Data
static class AnnexUploadDTO{
@FileNotEmpty(format = {"pdf","doc","zip"}, message = "文件为pdf/doc/zip格式")
private MultipartFile[] file;
}
}
结果展示
来源:https://blog.csdn.net/weixin_43931248/article/details/122000087


猜你喜欢
- For-Each循环For-Each循环也叫增强型的for循环,或者叫foreach循环。For-Each循环是JDK5.0的新特性(其他新
- 一、添加插件apply plugin: 'maven-publish'二、添加如下配置//打包源码task sourceJa
- 报错信息:java.lang.Exception: org.apache.hadoop.mapreduce.task.reduce.Shuf
- Maven是项目对象模型(POM),可以通过一小段描述信息来管理项目的构建,报告和文档的项目管理工具软件。Maven是一个项目管理工具,它包
- 免责声明:本教程所有资源均来源于网络;仅用于学习交流,请勿用于任何商业行为;如需要,请使用正版授权;侵权联删。推荐最新 IntelliJ I
- iOS定位 - 普通定位(没有地图) - 反地理编码(得到具体位置),下面通过代码给大家详解,代码如下:#import <CoreLo
- 1.Java内存模型JAVA定义了一套在多线程读写共享数据时时,对数据的可见性、有序性和原子性的规则和保障。屏蔽掉不同操作系统间的微小差异。
- 整理文档,搜刮出一个spring boot实现过滤器和 * demo ,稍微整理精简一下做下分享。 * 定义:@WebServletpubl
- 本文实例为大家分享了Qt实现计算器功能的具体代码,供大家参考,具体内容如下该计算器主要通过lineEdit获取和显示数字,通过tablevi
- 一、JVM 类加载机制JVM 类加载机制分为五个部分:加载,验证,准备,解析,初始化,下面我们就分别来看一下这五个过程。1. 加载:加载是类
- 服务端在平台上创建springboot小程序应用创建小程序登录蚂蚁金服开放平台,扫码登录填写信息后,点击支付宝小程序,选择立即接入 >
- 一、分步骤集成1.1 整合连接池hikariCP介绍:HikariCP 是一个高性能的 JDBC 连接池组件,可以避免连接频繁建立、关闭的开
- 一、概述System.Net.WebClient属于高层类、使用简单。均支持异步版本。支持http,https,fpt,files等URI。
- 在项目中经常要用到将字符串解析成Locale,但是没有一个比较好用的类。java本身提供了3个构造函数,但是实际使用过程中,需要自己解析,比
- 1、接口:一种把类抽象的更彻底,接口里只能包含抽象方法的“特殊类”。接口不关心类的内部状态数据,定义的是一批类所遵守的规范。(它只规定这批类
- 对于一个App的UI而言,在流畅性上的改进目标其实就是降低屏幕绘制的延迟,创建流畅和稳定的帧率以避免卡顿。在理想情况下,全部的测量、布局和绘
- JUC包(java.util.concurrent)中提供了对定时任务的支持,即ScheduledExecutorService接口。本文对
- 要将一个对象序列化,可是如果对象的属性为null的时候,我们想将属性为null的都去掉。在这里我使用Newtonsoft.Json.dll记
- Android强制异步转同步方法,供大家参考,具体内容如下Android系统中规定耗时任务需要在异步线程中进行,特别是网络请求必须在异步线程
- 本文实例分享了Android获取手机系统版本等信息的方法,供大家参考,具体内容如下:第一种代码:String phoneInfo = &qu