国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發 > Java > 正文

Spring Boot實現通用的接口參數校驗

2024-07-14 08:40:45
字體:
來源:轉載
供稿:網友

本文介紹基于 Spring Boot 和 JDK8 編寫一個 AOP ,結合自定義注解實現通用的接口參數校驗。

緣由

目前參數校驗常用的方法是在實體類上添加注解,但對于不同的方法,所應用的校驗規則也是不一樣的,例如有一個 AccountVO 實體:

public class AccountVO {  private String name; // 姓名  private Integer age; // 年齡}

假設存在這樣一個業務:用戶注冊時需要填寫姓名和年齡,用戶登陸時只需要填寫姓名就可以了。那么把校驗規則加在實體類上顯然就不合適了。

所以一直想實現一種方法級別的參數校驗,對于同一個實體參數,不同的方法可以應用不同的校驗規則,由此便誕生了這個工具,而且在日常工作中使用了很久。

介紹

先來看看使用的方式:

@Servicepublic class TestImpl implements ITestService {  @Override  @Check({"name", "age"})  public void testValid(AccountVO vo) {    // ...  }}

其中方法上的 @Check 注解指明了參數 AccountVO 中的 name 、 age 屬性不能為空。除了非空校驗外,還支持大小判斷、是否等于等校驗:

@Check({"id>=8", "name!=aaa", "title<10"})

默認的錯誤信息會返回字段,錯誤原因和調用的方法,例如:

updateUserId must not null while calling testValidid must >= 8 while calling testValidname must != aaa while calling testValid

也支持自定義錯誤返回信息:

@Check({"title<=8:標題字數不超過8個字,含標點符號"})public void testValid(TestPO po) {  // ...}

只需要在校驗規則后加上 : ,后面寫上自定義信息,就會替換默認的錯誤信息。

PS: 核心原理是通過反射獲取參數實體中的字段的值,然后根據規則進行校驗, 所以目前只支持含有一個參數的方法,并且參數不能是基礎類型。

使用

spring-boot 中如何使用 AOP 這里不再贅述,主要介紹 AOP 中的核心代碼。

Maven 依賴

除了 spring-boot 依賴之外,需要的第三方依賴,不是核心的依賴,可以根據個人習慣取舍:

<!-- 用于字符串校驗 --><dependency>  <groupId>org.apache.commons</groupId>  <artifactId>commons-lang3</artifactId>  <version>3.3.2</version></dependency><!-- 用于日志打印 --><dependency>  <groupId>org.slf4j</groupId>  <artifactId>slf4j-api</artifactId>  <version>1.7.25</version></dependency>

自定義注解

import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.Target;import static java.lang.annotation.RetentionPolicy.RUNTIME;/** * 參數校驗 注解 * Created by cipher on 2017/9/20. */@Target({ElementType.TYPE, ElementType.METHOD})@Retention(RUNTIME)public @interface Check {  // 字段校驗規則,格式:字段名+校驗規則+冒號+錯誤信息,例如:id<10:ID必須少于10  String[] value();}

核心代碼

通過切面攔截加上了 @Check 注解的接口方法,在方法執行前,執行參數校驗,如果存在錯誤信息,則直接返回:

@Around(value = "@com.cipher.checker.Check") // 這里要換成自定義注解的路徑public Object check(ProceedingJoinPoint point) throws Throwable {  Object obj;  // 參數校驗  String msg = doCheck(point);  if (!StringUtils.isEmpty(msg)) {    // 這里可以返回自己封裝的返回類    throw new IllegalArgumentException(msg);  }  obj = point.proceed();  return obj;}

核心的校驗方法在 doCheck 方法中,主要原理是獲取注解上指定的字段名稱和校驗規則,通過反射獲取參數實體中對應的字段的值,再進行校驗:

/** * 參數校驗 * * @param point ProceedingJoinPoint * @return 錯誤信息 */private String doCheck(ProceedingJoinPoint point) {  // 獲取方法參數值  Object[] arguments = point.getArgs();  // 獲取方法  Method method = getMethod(point);  String methodInfo = StringUtils.isEmpty(method.getName()) ? "" : " while calling " + method.getName();  String msg = "";  if (isCheck(method, arguments)) {    Check annotation = method.getAnnotation(Check.class);    String[] fields = annotation.value();    Object vo = arguments[0];    if (vo == null) {      msg = "param can not be null";    } else {      for (String field : fields) {        // 解析字段        FieldInfo info = resolveField(field, methodInfo);        // 獲取字段的值        Object value = ReflectionUtil.invokeGetter(vo, info.field);        // 執行校驗規則        Boolean isValid = info.optEnum.fun.apply(value, info.operatorNum);        msg = isValid ? msg : info.innerMsg;      }    }  }  return msg;}

可以看到主要的邏輯是:

解析字段 -> 獲取字段的值 -> 執行校驗規則

內部維護一個枚舉類,相關的校驗操作都在里面指定:

/** * 操作枚舉 */enum Operator {  /**   * 大于   */  GREATER_THAN(">", CheckParamAspect::isGreaterThan),  /**   * 大于等于   */  GREATER_THAN_EQUAL(">=", CheckParamAspect::isGreaterThanEqual),  /**   * 小于   */  LESS_THAN("<", CheckParamAspect::isLessThan),  /**   * 小于等于   */  LESS_THAN_EQUAL("<=", CheckParamAspect::isLessThanEqual),  /**   * 不等于   */  NOT_EQUAL("!=", CheckParamAspect::isNotEqual),  /**   * 不為空   */  NOT_NULL("not null", CheckParamAspect::isNotNull);  private String value;  private BiFunction<Object, String, Boolean> fun;  Operator(String value, BiFunction<Object, String, Boolean> fun) {    this.value = value;    this.fun = fun;  }}

由于篇幅原因,這里就不一一展開所有的代碼,有興趣的朋友可以到以下地址獲取所有的源碼: ciphermagic/java-learn/sandbox/checker

TODO

  1. 以Spring Boot Starter的方式封裝成獨立組件
  2.  支持正則表達式驗證

最后

感謝大家的閱讀,喜歡的朋友可以在github上點個贊,有任何問題或者建議請在下方留言,期待你的回復。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 吉水县| 日土县| 新沂市| 凤翔县| 阆中市| 孙吴县| 嘉禾县| 连云港市| 旌德县| 玉田县| 疏勒县| 望都县| 康平县| 泸定县| 焦作市| 靖西县| 贡觉县| 阳新县| 凤山县| 务川| 绩溪县| 北碚区| 秦皇岛市| 聂荣县| 休宁县| 留坝县| 陈巴尔虎旗| 金平| 衡山县| 平武县| 灵台县| 万荣县| 莫力| 新晃| 军事| 稻城县| 南靖县| 馆陶县| 化州市| 将乐县| 隆林|