在实际开发中,常常会对接口api入参做格式验证,有些入参是List数组对象。
public ApiResponse receiveManualTranscoding(@Validated @RequestBody List<SdZmDto> sdZmDtos)
其实这样是失效的,会发现其SdZmDto并不会进行验证。
因为@Validated参数只能验证单个对象。
解决方法1
手动对每一个参数对象进行做验证,但是代码会显得冗余
private ResultVo validate(@Valid SdZmDto gwbaDto) {
Set<ConstraintViolation<@Valid SdZmDto >> validateSet = Validation.buildDefaultValidatorFactory()
.getValidator()
.validate(gwbaDto);
if (!CollectionUtils.isEmpty(validateSet)) {
String messages = validateSet.stream()
.map(ConstraintViolation::getMessage)
.reduce((m1, m2) -> m1 + ";" + m2)
.orElse("参数输入有误!");
return ResultVo.builder().code(ResultConstants.FALSE).message(SYSTEM_ERROR_TSBAMESSAGE + messages).build();
}
return null;
}
解决方法2
定义一个对象,将list封装进行,则验证就验证单个对象了。
@Data
public class ValidList<E> implements List<E> {
@Valid
private List<E> list = new ArrayList<>();
}
将 public ApiResponse receiveManualTranscoding(@Validated @RequestBody List<SdZmDto> sdZmDtos)
改成
public ApiResponse receiveManualTranscoding(@Validated @RequestBody ValidList<SdZmDto> sdZmDtos)
选择方案
解决方法1虽然冗余代码,但是符合需求,如果采用方法2,则需要将请求的json参数进行做调整。
@PostMapping("/receiveManualTranscoding")
@ResponseBody
public ApiResponse receiveManualTranscoding(@RequestBody List<SdZmDto> sdZmDtos) {
for(SdZmDto sdZmDto : sdZmDtos){
ApiResponse apiResponse = null;
if((apiResponse = validate(sdZmDto)) != null){
return apiResponse;
}
}
ajmApiService.receiveManualTranscoding(sdZmDtos);
return ApiDataResponse.ofSuccess();
}
private ApiResponse validate(@Valid SdZmDto sdZmDto) {
Set<ConstraintViolation<@Valid SdZmDto >> validateSet = Validation.buildDefaultValidatorFactory()
.getValidator()
.validate(sdZmDto);
if (!CollectionUtils.isEmpty(validateSet)) {
String messages = validateSet.stream()
.map(ConstraintViolation::getMessage)
.reduce((m1, m2) -> m1 + ";" + m2)
.orElse("参数输入有误!");
return ApiResponse.ofFail(messages);
}
return null;
}
```