手把手寫一個基於Spring Boot框架下的參數校驗組件(JSR-303)

来源:https://www.cnblogs.com/zdd-java/archive/2020/01/29/12240929.html
-Advertisement-
Play Games

前言 之前參與的新開放平臺研發的過程中,由於不同的介面需要對不同的入參進行校驗,這就涉及到通用參數的校驗封裝,如果不進行封裝,那麼寫出來的校驗代碼將會風格不統一、校驗工具類不一致、維護風險高等其它因素,於是我對其公共的校驗做了一個封裝,達到了通過註解的方式即可實現參數統一校驗。 遇到的問題 在封裝的 ...


前言  

        之前參與的新開放平臺研發的過程中,由於不同的介面需要對不同的入參進行校驗,這就涉及到通用參數的校驗封裝,如果不進行封裝,那麼寫出來的校驗代碼將會風格不統一、校驗工具類不一致、維護風險高等其它因素,於是我對其公共的校驗做了一個封裝,達到了通過註解的方式即可實現參數統一校驗。

遇到的問題

                      在封裝的時候就發現了一個問題,我們是開放平臺,返回的報文都必須是統一風格,也就是類似於{code:999,msg:"參數校驗失敗",data:null} 這種,但是原生的JSR303並不支持自定義的欄位,所以需要自定義校驗註解。針對這個問題我參考一些JSR303的資料,對其進行了一個定製擴展,以達到開發人員不需要關註捕捉和封裝返回信息。

  

傳統的校驗做法 

        如下校驗如果一個實體裡面上百個欄位需要校驗的話,對於維護起來是一個很麻煩的事情,而且很多校驗可以通過jsr-303的註解方式統一處理,無需寫一大堆if和else

 if(name == null) {
     //返回錯誤信息
 }else if(age == null) {
     //返回錯誤信息
}

 基於jsr-303定製後的校驗

  1. 定義一個自定義非空註解
@Target({ METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER })
@Retention(RUNTIME)
@Documented
@Constraint(validatedBy = { CorpNotEmptyValidator.class })
public @interface CorpNotEmpty {

    //自定義欄位
    String field() default  "";
   //返回錯誤碼
    String code() default  "0";
    //錯誤消息
    String message() default "{javax.validation.constraints.NotNull.message}";

    Class<?>[] groups() default { };

    Class<? extends Payload>[] payload() default { };

    @Target({ METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER })
    @Retention(RUNTIME)
    @Documented
    @interface List {
        CorpNotEmpty[] value();
    }
}

   2.定義非空註解對應的校驗器, initialize和isValid作用描述如下:

  •   initialize方法主要是初始化ReturnCodeModel,用於當校驗參數不通過後返回,ReturnCodeModel裡面主要是封裝了返回體,如 code,message等
  •        isValid主要是自定義校驗器的校驗規則,如下判斷是否為空使用 StringUtils.isEmpty方法,如果校驗不通過則set flag為false,然後調用基類的isValid方法,該基類方法會判斷flag是否為false,如果是false說明不通過
public class CorpNotEmptyValidator extends BaseCorpValidator<CorpNotEmpty,String> {
    @Override
    public void initialize(CorpNotEmpty constraintAnnotation) {
        model = new ReturnCodeModel();
        model.setCode(constraintAnnotation.code());
        model.setErrorMsg(constraintAnnotation.message());
        model.setField(constraintAnnotation.field());
    }
 
    @Override
    public boolean isValid(String s, ConstraintValidatorContext constraintValidatorContext) {
        System.out.println("1");
        if(StringUtils.isEmpty(s)){
            model.setFlag(false);
        }else{
            model.setFlag(true);
        }
        return super.isValid(s,constraintValidatorContext);
    }
}

  3.定義一個基類,實現 ConstraintValidator,主要是因為需要把isValid這個方法定義成抽象方法提供給不同的校驗器使用,避免其它校驗器寫重覆的代碼

public abstract class BaseCorpValidator<A extends Annotation,B> implements ConstraintValidator<A ,B> {
    protected ReturnCodeModel model = null;
    @Override
    public boolean isValid(B value, ConstraintValidatorContext context) {
        if(!model.getFlag()){
            context.disableDefaultConstraintViolation();
            context.buildConstraintViolationWithTemplate(JSON.toJSONString(model)).addConstraintViolation();
            return false;
        }
        return true;
    }

}

   4.測試類

public class TestV1 {
    static ValidatorFactory validatorFactory = Validation.buildDefaultValidatorFactory();
    static Validator validator = validatorFactory.getValidator();
    public static void main(String[] args) {
        UserModel userModel = new UserModel();
        userModel.setName("aa");
        userModel.setDate("2011");
        Set<ConstraintViolation<UserModel>> constraintViolations  = validator.validate(userModel);
        //判斷constraintViolations是否為空,不為空說明校驗不通過,拿到ReturnCodeModel裡面的錯誤信息後返回給客戶端
        if(!constraintViolations.isEmpty()){
            for (ConstraintViolation<?> item : constraintViolations) {
                ReturnCodeModel codeModel = JSON.parseObject(item.getMessage(), ReturnCodeModel.class);
                System.out.println(JSON.toJSONString(codeModel));
            }
        }
    }
}

 

畫外音:場景考慮

  1.比如name這個欄位,要滿足既不能為空又只能為數字這2個情況,如果把2個校驗方法都寫在同一個校驗器,則其他開發使用的時候也會影響到,所以需要有2個註解的方式,一個是校驗為空,一個是校驗是否位數字,分析完後那麼就存在一個先後順序問題(因為自己在本地測試出現有可能會先執行校驗是否位數字的校驗器,這時候就會出現空指針異常), 所以針對這個場景需要自定義一個順序註解。

  如下代碼,在需要校驗的model實體上加入@GroupSequence註解,這樣校驗器底層會幫我們按照順序依次處理

//順序註解
@GroupSequence({
    First.class,
    Two.class,
    Three.class,
    UserModel.class
})
public class UserModel {
	@CorpMustNumber(code="-2",message="必須數字",groups=Two.class)//在執行數字校驗
	@CorpNotEmpty(code="-1",message="姓名必填",groups=First.class)//先執行非空
	private String name;
	@CorpNotEmpty(code="-1",message="日期必填")
	private String date;
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public String getDate() {
		return date;
	}
	public void setDate(String date) {
		this.date = date;
	}
	
	
}

  

First Two

總結

    以上就是本篇博客涉及到技術點的所有代碼,通過定製自己的校驗器以滿足公司業務場景,對於開發來說統一了規範,統一風格,對以後維護還是擴展都非常方便。 如果博文對你有幫助麻煩點個關註或者贊,謝謝!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • HandlerMapping 處理器映射 HTTP請求被DispatcherServlet攔截後,會調用HandlerMapping來處理,HandlerMapping根據 url<=>controller 之間的映射關係來確定要調用哪個controller來處理。 有2種HandlerMappin ...
  • 結論 RestTemplate 的 postForObject 方法有四個參數 String url = 顧名思義 這個參數是請求的url路徑 Object request = 請求的body 這個參數需要再controller類用 @RequestBody 註解接收 Class responseT ...
  • 錯誤 這邊調用的時候使用了RestTemplate 使用過程 下麵是我請求的路徑 調用之後程式報以下錯誤 解決方案: 我們繼承 MappingJackson2HttpMessageConverter 併在構造過程中設置其支持的 MediaType 類型即可: 然後把這個 WxMappingJacks ...
  • spring boot後臺時間正確,返回給前臺的時間不正確,和後臺差8個小時 原因是: spring boot中對於@RestController或者@Controller+@ResponseBody註解的介面方法的返回值預設是Json格式, 所以當對於date類型的數據,在返回瀏覽器端是會被spr ...
  • 預設是單例 通過註解@Scope("prototype"),將其設置為多例模式 參考: 曾經面試的時候有面試官問我spring的controller是單例還是多例,結果我傻逼的回答當然是多例,要不然controller類中的非靜態變數如何保證是線程安全的,這樣想起似乎是對的,但是不知道(主要是我沒看 ...
  • 字元串 字元串字面量:就是指這個字元串本身,比如"Java","Hello"。 字元串對象:比如new String("abc"),或者直接String s="str",後面的"str"也是一個字元串對象。 字元串引用:引用就是一個變數,指向對應的字元串對象。 常量池 class常量池 Java源文 ...
  • vue方法 java方法 ...
  • 1、構造需要從字典構造cds={'code':["002372.XSHE","002415.XSHE","002304.XSHE","600519.XSHG","600196.XSHG"], #代碼 'name':["偉星新材", "海康威視", "洋河股份", "貴州茅臺", "復星醫葯"]} c... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...