詳解Spring AOP 實(shí)現(xiàn)“切面式”valid校驗(yàn)
why:
為什么要用aop實(shí)現(xiàn)校驗(yàn)?
answer:
spring mvc 默認(rèn)自帶的校驗(yàn)機(jī)制 @Valid + BindingResult, 但這種默認(rèn)實(shí)現(xiàn)都得在Controller方法的中去接收BindingResult,從而進(jìn)行校驗(yàn).
eg:
if (result.hasErrors()) {
List<ObjectError> allErrors = result.getAllErrors();
List<String> errorlists = new ArrayList<>();
for (ObjectError objectError : allErrors) {
errorlists.add(objectError.getDefaultMessage());
}
}
獲取errorlists。這樣實(shí)現(xiàn)的話,每個(gè)需要校驗(yàn)的方法都得重復(fù)調(diào)用,即使封裝也是。
可能上面那么說(shuō)還不能表明spring 的@Valid + BindingResult實(shí)現(xiàn),我先舉個(gè)“栗子”。
1. 栗子(舊版本)
1.1 接口層(IDAL)
eg: 簡(jiǎn)單的POST請(qǐng)求,@RequestBody接收請(qǐng)求數(shù)據(jù),@Valid + BindingResult進(jìn)行校驗(yàn)
- httpMethid: POST
- parameters:@RequestBody接收請(qǐng)求數(shù)據(jù)
- valid:@Valid +BindingResult
@ResponseBody
@PostMapping("body")
public ResponseVO bodyPost(@RequestBody @Valid TestVO body,BindingResult result){
//校驗(yàn)到錯(cuò)誤
if (result.hasErrors()) {
List<ObjectError> allErrors = result.getAllErrors();
List<String> lists = new ArrayList<>();
for (ObjectError objectError : allErrors) {
lists.add(objectError.getDefaultMessage());
}
return new ResponseVO(HttpStatus.BAD_REQUEST.value(), "parameter empty", lists);
}
return new ResponseVO(HttpStatus.OK.value(), "bodyPost", null);
}
1.2 實(shí)體(vo)校驗(yàn)內(nèi)容
@Valid + BindingResult的校驗(yàn)注解一大堆,網(wǎng)上一摸就有的!
public class TestVO {
@Getter
@Setter
@Min(value = 0,message = "請(qǐng)求參數(shù)isString不能小于0")
private Integer isInt;
@Getter
@Setter
@NotBlank(message = "請(qǐng)求參數(shù)isString不能為空")
private String isString;
}
1.3 結(jié)果測(cè)試

2. aop校驗(yàn)(升級(jí)版)
可以看到若是多個(gè)像bodyPost一樣都需要對(duì)body進(jìn)行校驗(yàn)的話,那么有一坨代碼就必須不斷復(fù)現(xiàn),即使改為父類可復(fù)用方法,也得去調(diào)用。所以左思右想還是覺得不優(yōu)雅。所以有了aop進(jìn)行切面校驗(yàn)。
2.1 接口層(IDAL)
是的!你沒看錯(cuò),上面那一坨代碼沒了,也不需要調(diào)用父類的的共用方法。就單單一個(gè)注解就完事了:@ParamValid
@ParamValid
@ResponseBody
@PostMapping("body")
public ResponseVO bodyPost(@RequestBody @Valid TestVO body,BindingResult result){
return new ResponseVO("bodyPost", null);
}
2.2 自定義注解(annotation)
這個(gè)注解也是簡(jiǎn)簡(jiǎn)單單的用于方法的注解。
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ParamValid {}
2.3 重點(diǎn)!切面實(shí)現(xiàn)(Aspect)
切面詳解:
@Before: 使用注解方式@annotation(XX),凡是使用到所需切的注解(@ParamValid),都會(huì)調(diào)用該方法
JoinPoint: 通過(guò)JoinPoint獲取方法的參數(shù),以此獲取BindingResult所校驗(yàn)到的內(nèi)容
遷移校驗(yàn)封裝: 將原先那一坨校驗(yàn)遷移到Aspect中:validRequestParams
響應(yīng)校驗(yàn)結(jié)果:
- 通過(guò)RequestContextHolder獲取response
- 獲取響應(yīng)OutputStream
- 將BindingResult封裝響應(yīng)
@Aspect
@Component
public class ParamValidAspect {
private static final Logger log = LoggerFactory.getLogger(ParamValidAspect.class);
@Before("@annotation(paramValid)")
public void paramValid(JoinPoint point, ParamValid paramValid) {
Object[] paramObj = point.getArgs();
if (paramObj.length > 0) {
if (paramObj[1] instanceof BindingResult) {
BindingResult result = (BindingResult) paramObj[1];
ResponseVO errorMap = this.validRequestParams(result);
if (errorMap != null) {
ServletRequestAttributes res = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletResponse response = res.getResponse();
response.setCharacterEncoding("UTF-8");
response.setContentType(MediaType.APPLICATION_JSON_UTF8_VALUE);
response.setStatus(HttpStatus.BAD_REQUEST.value());
OutputStream output = null;
try {
output = response.getOutputStream();
errorMap.setCode(null);
String error = new Gson().toJson(errorMap);
log.info("aop 檢測(cè)到參數(shù)不規(guī)范" + error);
output.write(error.getBytes("UTF-8"));
} catch (IOException e) {
log.error(e.getMessage());
} finally {
try {
if (output != null) {
output.close();
}
} catch (IOException e) {
log.error(e.getMessage());
}
}
}
}
}
}
/**
* 校驗(yàn)
*/
private ResponseVO validRequestParams(BindingResult result) {
if (result.hasErrors()) {
List<ObjectError> allErrors = result.getAllErrors();
List<String> lists = new ArrayList<>();
for (ObjectError objectError : allErrors) {
lists.add(objectError.getDefaultMessage());
}
return new ResponseVO(HttpStatus.BAD_REQUEST.value(), "parameter empty", lists);
}
return null;
}
}
2.4 測(cè)試結(jié)果

看了上面兩種結(jié)果,就可以對(duì)比出使用Spring AOP 配合@Valid + BindingResult進(jìn)行校驗(yàn)的優(yōu)點(diǎn):
- 去除代碼冗余
- AOP異步處理
- 優(yōu)化代碼實(shí)現(xiàn)
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot 項(xiàng)目添加 MDC 日志鏈路追蹤的執(zhí)行流程
日志鏈路追蹤就是將一個(gè)標(biāo)志跨線程進(jìn)行傳遞,在一般的小項(xiàng)目中也就是在你新起一個(gè)線程的時(shí)候,或者使用線程池執(zhí)行任務(wù)的時(shí)候會(huì)用到,比如追蹤一個(gè)用戶請(qǐng)求的完整執(zhí)行流程,本文給大家介紹SpringBoot MDC 日志鏈路追蹤的代碼,感興趣的朋友一起看看吧2021-06-06
Curator實(shí)現(xiàn)zookeeper的節(jié)點(diǎn)監(jiān)聽詳解
這篇文章主要介紹了Curator實(shí)現(xiàn)zookeeper的節(jié)點(diǎn)監(jiān)聽詳解,Curtor框架中一共有三個(gè)實(shí)現(xiàn)監(jiān)聽的方式,一種是NodeCache監(jiān)聽指定節(jié)點(diǎn),一種是pathChildrenCache監(jiān)聽子節(jié)點(diǎn),一種是TreeCache可以監(jiān)控所有節(jié)點(diǎn) 相當(dāng)于以上兩種的合集,需要的朋友可以參考下2023-12-12
關(guān)于IDEA配置Hibernate中遇到的問(wèn)題解決
這篇文章主要給大家介紹了關(guān)于IDEA配置Hibernate中遇到的問(wèn)題,文中通過(guò)圖文介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05
Spring學(xué)習(xí)通過(guò)AspectJ注解方式實(shí)現(xiàn)AOP操作
這篇文章主要為大家介紹了Spring學(xué)習(xí)通過(guò)AspectJ注解方式實(shí)現(xiàn)AOP操作,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05
SpringMVC中處理靜態(tài)資源的過(guò)程詳解
本文給大家介紹SpringMVC中處理靜態(tài)資源的過(guò)程,結(jié)合示例代碼給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2023-11-11
SpringMVC中的DispatcherServlet詳細(xì)解析
這篇文章主要介紹了SpringMVC中的DispatcherServlet詳細(xì)解析,DispatcherServlet也是一個(gè)Servlet,它也能通過(guò)Servlet的API來(lái)響應(yīng)請(qǐng)求,從而成為一個(gè)前端控制器,Web容器會(huì)調(diào)用Servlet的doGet()以及doPost()等方法,需要的朋友可以參考下2023-12-12
使用Spring初始化加載InitializingBean()方法
這篇文章主要介紹了使用Spring初始化加載InitializingBean()方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01

