spring boot+自定義 AOP 實現全域性校驗

吃桃子的小松鼠發表於2019-04-09

最近公司重構專案,重構為最熱的微服務框架 spring boot, 重構的時候遇到幾個可以統一處理的問題,也是專案中經常遇到,列如:統一校驗引數,統一捕獲異常。。。

僅憑程式碼 去控制引數的校驗,有時候是冗餘的,但通過框架支援的 去控制引數的校驗,是對於開發者很友好,先看下面的例子

1     @NotEmpty(message="手機號不能為空")
2     @Size(min=11,max=11,message="手機號碼長度不正確")
3     @Pattern(regexp=StringUtils.REGEXP_MOBILE,message="手機號格式不正確")
4     private String mobile;
View Code

這是spring boot支援的 校驗註解,然後我們在 contoller層 加上@Valid 註解 就可以達到校驗的目的。這是一種框架自帶的

本章 就展示一種 自定義的 AOP 校驗,首先 寫一個註解,註解裡面可以寫上 我們需要校驗的規則, 比如長度,正則。。。

 1 @Documented
 2 @Target({ElementType.FIELD,ElementType.METHOD})
 3 @Retention(RetentionPolicy.RUNTIME)
 4 public @interface ValidateParam {
 5 
 6     int min() default 0;
 7     
 8     int max() default Integer.MAX_VALUE;
 9     
10     String message() default "params is not null";
11      
12     String regexp();
13     
14     Class<?>[] groups() default { };
15 
16      Class<? extends Payload>[] payload() default { };
17      
18      boolean isNotNull() default true;
19 
20 }
View Code

然後定義一個AOP類

  1 package com.onecard.primecard.common.aop;
  2 
  3 import java.lang.reflect.Field;
  4 import java.lang.reflect.Method;
  5 import java.lang.reflect.ParameterizedType;
  6 import java.util.ArrayList;
  7 import java.util.Arrays;
  8 import java.util.regex.Pattern;
  9 
 10 import org.aspectj.lang.JoinPoint;
 11 import org.aspectj.lang.ProceedingJoinPoint;
 12 import org.aspectj.lang.annotation.Around;
 13 import org.aspectj.lang.annotation.Aspect;
 14 import org.aspectj.lang.annotation.Before;
 15 import org.aspectj.lang.annotation.Pointcut;
 16 import org.slf4j.Logger;
 17 import org.slf4j.LoggerFactory;
 18 import org.springframework.context.ApplicationContext;
 19 import org.springframework.context.support.ClassPathXmlApplicationContext;
 20 import org.springframework.stereotype.Component;
 21 
 22 import com.jfcf.core.dto.ResultData;
 23 import com.onecard.core.support.util.StringUtils;
 24 import com.onecard.primecard.common.annotation.ValidateParam;
 25 import com.onecard.primecard.common.utils.ResultDataUtil;
 26 
 27 
 28 
 29 
 30 /**
 31  * 全域性 切面類(校驗引數)
 32  * 
 33  * @author Administrator
 34  *
 35  */
 36 @Aspect
 37 @Component
 38 public class GobalHandlerAspect {
 39     
 40     private static Logger logger = LoggerFactory.getLogger(GobalHandlerAspect.class);
 41     
 42     @Pointcut("execution(* 包名.controller..*.*(..)) && execution(* 包名.controller..*.*(..))")
 43     public void checkAspect(){};
 44     
 45     @Before("checkAspect()")
 46     public void befor(JoinPoint joinPoint) throws Exception{
 47         //前置統一輸出引數
 48         Object[] args = joinPoint.getArgs();
 49         if(args != null && args.length>0){
 50             Object obj = args[0];
 51             ParameterizedType pt = (ParameterizedType)obj.getClass().getGenericSuperclass();
 52             Class<?> classzz = (Class<?>) pt.getActualTypeArguments()[0];
 53             logger.info("【小X卡】-【請求實體入參】:"+classzz.newInstance().toString());
 54         }
 55         
 56     }
 57     
 58     @Around("checkAspect()")
 59     public Object around(ProceedingJoinPoint joinPoint) throws Throwable{
 60         //校驗引數
 61         Object[] args = joinPoint.getArgs();
 62         Object obj = null;
 63         if(args != null && args.length > 0){
 64             obj = args[0];
 65             Class classzz = obj.getClass();
 66             //沒有順序和秩序的陣列
 67             Field[] fieldArray = classzz.getDeclaredFields();
 68             ArrayList<Field> fieldList = new ArrayList<Field>(Arrays.asList(fieldArray));
 69             String res = checkParam(fieldList,obj);
 70             if(StringUtils.isNotNull(res)){
 71                 return ResultDataUtil.result(ResultData.STATUS_PARAM_ERROR, res);
 72             }
 73         }
 74         
 75         return joinPoint.proceed();
 76     }
 77 
 78     private String checkParam(ArrayList<Field> fieldList, Object obj) throws Exception {
 79         
 80         for(Field field : fieldList){
 81             ValidateParam validateParam = field.getAnnotation(ValidateParam.class);
 82             logger.info("【小X卡】獲取註解值:"+validateParam.isNotNull()+"min="+validateParam.min()+"max="+validateParam.max());
 83             Method method = obj.getClass().getMethod("get"+getMethodName(field.getName()));
 84             logger.info("【小X卡】入參實體方法名稱:"+method.getName());
 85             if(method != null){
 86                 Object val = method.invoke(obj);
 87                 logger.info("【小x卡】回撥方法:"+val);
 88                 if(validateParam != null && validateParam.isNotNull() == true){
 89                     if(null == val || "".equals(val) ){
 90                         return field.getName()+"必填引數為空";
 91                     }
 92                 }
 93                 if(validateParam.min()==11 && validateParam.max() == 11){
 94                     if(val.toString().length() != 11){
 95                         return field.getName()+"請輸入引數正確的長度";
 96                     }
 97                 }
 98                 if(validateParam.regexp().equals(StringUtils.REGEXP_MOBILE)){
 99                     if(!Pattern.matches(StringUtils.REGEXP_MOBILE, val.toString())){
100                         return field.getName()+"引數格式錯誤";
101                     }
102                 }
103             }
104         }
105         return null;
106     }
107     
108     
109      /**
110      * 方法首字母大寫
111      * @param fieldName
112      * @return
113      */
114     private String getMethodName(String fieldName) {
115         StringBuffer buffer = new StringBuffer();
116         String firstLetter = fieldName.substring(0, 1).toUpperCase();
117         return buffer.append(firstLetter).append(fieldName.substring(1, fieldName.length())).toString();
118 
119     }       
120  }
View Code

定義一個切點 @Pointcut, 用execution 表示式,去獲取要校驗的 某個類 和某個方法, 也就是連線點,然後 用定義一個通知,上面程式碼中有2個通知,一個前置通知@Before,一個環繞通知@Around,我們使用功能最強大的環繞通知。

通過上面的程式碼可以看出  首先獲取引數,然後通過反射機制 獲取 入參物件中的全部欄位, 再去獲取 我們在欄位中加 我們自定義註解的欄位,通過反射方法的回撥,獲取欄位值,對值做判斷, 返回校驗結果。

 

相關文章