最近公司重構專案,重構為最熱的微服務框架 spring boot, 重構的時候遇到幾個可以統一處理的問題,也是專案中經常遇到,列如:統一校驗引數,統一捕獲異常。。。
僅憑程式碼 去控制引數的校驗,有時候是冗餘的,但通過框架支援的 去控制引數的校驗,是對於開發者很友好,先看下面的例子
1 @NotEmpty(message="手機號不能為空") 2 @Size(min=11,max=11,message="手機號碼長度不正確") 3 @Pattern(regexp=StringUtils.REGEXP_MOBILE,message="手機號格式不正確") 4 private String mobile;
這是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 }
然後定義一個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 }
定義一個切點 @Pointcut, 用execution 表示式,去獲取要校驗的 某個類 和某個方法, 也就是連線點,然後 用定義一個通知,上面程式碼中有2個通知,一個前置通知@Before,一個環繞通知@Around,我們使用功能最強大的環繞通知。
通過上面的程式碼可以看出 首先獲取引數,然後通過反射機制 獲取 入參物件中的全部欄位, 再去獲取 我們在欄位中加 我們自定義註解的欄位,通過反射方法的回撥,獲取欄位值,對值做判斷, 返回校驗結果。