用好spring mvc validator可以簡化代碼

来源:http://www.cnblogs.com/ASPNET2008/archive/2016/09/01/5831766.html
-Advertisement-
Play Games

表單的數據檢驗對一個程式來講非常重要,因為對於客戶端的數據不能完全信任,常規的檢驗類型有: 參數為空,根據不同的業務規定要求表單項是必填項 參數值的有效性,比如產品的價格,一定不能是負數 多個表單項組合檢驗,比如在註冊時密碼與確認密碼必須相同 參數值的數據範圍,常見的是一些狀態值,或者叫枚舉值,如果 ...


表單的數據檢驗對一個程式來講非常重要,因為對於客戶端的數據不能完全信任,常規的檢驗類型有:

  • 參數為空,根據不同的業務規定要求表單項是必填項
  • 參數值的有效性,比如產品的價格,一定不能是負數
  • 多個表單項組合檢驗,比如在註冊時密碼與確認密碼必須相同
  • 參數值的數據範圍,常見的是一些狀態值,或者叫枚舉值,如果傳遞的參數超出已經定義的枚舉那麼也是無意義的

上面的這些檢驗基本上都是純數據方面的,還不算具體的業務數據檢驗,下麵是一些強業務相關的數據檢驗

  • 根據產品ID,去檢驗ID是否真實存在
  • 註冊用戶時,需要檢驗用戶名的唯一性
  • ....


根據上面的需求,如果我們將這些檢驗的邏輯全部與業務邏輯耦合在一起,那麼我們的程式邏輯將會變得冗長而且不便於代碼復用,下麵的代碼就是耦合性強的一種體現:

           if (isvUserRequestDTO == null) {
                log.error("can not find isv request by request id, " + isvRequestId);
                return return_value_error(ErrorDef.FailFindIsv);
            }
            if (isvUserRequestDTO.getAuditStatus() != 1) {
                log.error("isv request is not audited, " + isvRequestId);
                return return_value_error(ErrorDef.IsvRequestNotAudited);
            }

 

我們可以利用spring提供的validator來解耦表單數據的檢驗邏輯,可以將上述的代碼從具體的業務代碼的抽離出去。

 


Hibernate validator,它是JSR-303的一種具體實現。它是基於註解形式的,我們看一下它原生支持的一些註解。

註解 說明
@Null 只能為空,這個用途場景比較少
@NotNull 不能為空,常用註解
@AssertFalse 必須為false,類似於常量
@AssertTrue 必須為true,類似於常量
@DecimalMax(value)  
@DecimalMin(value)  
@Digits(integer,fraction)  
@Future 代表是一個將來的時間
@Max(value) 最大值,用於一個枚舉值的數據範圍控制
@Min(value) 最小值,用於一個枚舉值的數據範圍控制
@Past 代表是一個過期的時間
@Pattern(value) 正則表達式,比如驗證手機號,郵箱等,非常常用
@Size(max,min)

限制字元長度必須在min到max之間

 

 

 

 

 

 

 

 

 

 

 

 

 

 基礎數據類型的使用示例

@NotNull(message = "基礎數量不能為空")
    @Min(value = 0,message = "基礎數量不合法")
    private Integer baseQty;


嵌套檢驗,如果一個對象中包含子對象(非基礎數據類型)需要在屬性上增加@Valid註解。

   @Valid
   @NotNull(message = "價格策略內容不能為空")
    private List<ProductPricePolicyItem> policyItems;


除了原生提供的註解外,我們還可以自定義一些限制性的檢驗類型,比如上面提到的多個屬性之間的聯合檢驗。該註解需要使用@Constraint標註,這裡我編寫了一個用於針對兩個屬性之間的數據檢驗的規則,它支持兩個屬性之間的如下操作符,而且可以設置多組屬性對。

  • ==
  • >
  • >=
  • <
  • <=

創建註解

  • 通過@Constraint指定檢驗的實現類CrossFieldMatchValidator
  • 增加兩個屬性名稱欄位,用於後續的檢驗
  • 增加一個註解的List,用來支持一個對象中檢驗多組屬性對。比如即需要檢驗最大數量與最小數量,也需要檢驗密碼與確認密碼
@Target({ElementType.TYPE, ElementType.ANNOTATION_TYPE})
@Retention(RUNTIME)
@Constraint(validatedBy = CrossFieldMatchValidator.class)
@Documented
public @interface CrossFieldMatch {

    String message() default "{constraints.crossfieldmatch}";

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

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

    /**
     * @return The first field
     */
    String first();

    /**
     * @return The second field
     */
    String second();

    /**
     * first operator second
     * @return
     */
    CrossFieldOperator operator();

    /**
     * Defines several <code>@FieldMatch</code> annotations on the same element
     *
     * @see CrossFieldMatch
     */
    @Target({ElementType.TYPE, ElementType.ANNOTATION_TYPE})
    @Retention(RUNTIME)
    @Documented
    @interface List {
        CrossFieldMatch[] value();
    }
}

檢驗實現類

isValid方法,通過反射可以取到需要檢驗的兩個欄位的值以及數據類型,然後根據指定的數據操作符以及數據類型做出計算。目前這個檢驗只針對我的業務並不十分通用,需要根據自己的項目情況來靈活處理。

public class CrossFieldMatchValidator implements ConstraintValidator<CrossFieldMatch, Object> {

    private String firstFieldName;
    private String secondFieldName;
    private CrossFieldOperator operator;

    @Override
    public void initialize(final CrossFieldMatch constraintAnnotation) {
        firstFieldName = constraintAnnotation.first();
        secondFieldName = constraintAnnotation.second();
        operator=constraintAnnotation.operator();
    }

    @Override
    public boolean isValid(final Object value, final ConstraintValidatorContext context) {
        try {
            Class valueClass=value.getClass();
            final Field firstField = valueClass.getDeclaredField(firstFieldName);
            final Field secondField = valueClass.getDeclaredField(secondFieldName);
            //不支持為null的欄位
            if(null==firstField||null==secondField){
                return false;
            }

            firstField.setAccessible(true);
            secondField.setAccessible(true);
            Object firstFieldValue= firstField.get(value);
            Object secondFieldValue= secondField.get(value);

            //不支持類型不同的欄位
            if(!firstFieldValue.getClass().equals(secondFieldValue.getClass())){
                return false;
            }

            //整數支持 long int short
            //浮點數支持 double
            if(operator==CrossFieldOperator.EQ) {
                return firstFieldValue.equals(secondFieldValue);
            }
            else if(operator==CrossFieldOperator.GT){
                if(firstFieldValue.getClass().equals(Long.class)||firstFieldValue.getClass().equals(Integer.class)||firstFieldValue.getClass().equals(Short.class)) {
                    return (Long)firstFieldValue > (Long) secondFieldValue;
                }
                else if(firstFieldValue.getClass().equals(Double.class)) {
                    return (Double)firstFieldValue > (Double) secondFieldValue;
                }

            }
            else if(operator==CrossFieldOperator.GE){
                if(firstFieldValue.getClass().equals(Long.class)||firstFieldValue.getClass().equals(Integer.class)||firstFieldValue.getClass().equals(Short.class)) {
                    return Long.valueOf(firstFieldValue.toString()) >= Long.valueOf(secondFieldValue.toString());
                }
                else if(firstFieldValue.getClass().equals(Double.class)) {
                    return Double.valueOf(firstFieldValue.toString()) >= Double.valueOf(secondFieldValue.toString());
                }
            }
            else if(operator==CrossFieldOperator.LT){
                if(firstFieldValue.getClass().equals(Long.class)||firstFieldValue.getClass().equals(Integer.class)||firstFieldValue.getClass().equals(Short.class)) {
                    return (Long)firstFieldValue < (Long) secondFieldValue;
                }
                else if(firstFieldValue.getClass().equals(Double.class)) {
                    return (Double)firstFieldValue < (Double) secondFieldValue;
                }
            }
            else if(operator==CrossFieldOperator.LE){
                if(firstFieldValue.getClass().equals(Long.class)||firstFieldValue.getClass().equals(Integer.class)||firstFieldValue.getClass().equals(Short.class)) {
                    return Long.valueOf(firstFieldValue.toString()) <= Long.valueOf(secondFieldValue.toString());
                }
                else if(firstFieldValue.getClass().equals(Double.class)) {
                    return Double.valueOf(firstFieldValue.toString()) <= Double.valueOf(secondFieldValue.toString());
                }
            }
        }
        catch (final Exception ignore) {
            // ignore
        }
        return false;
    }
}

 

調用示例:

@CrossFieldMatch.List({
        @CrossFieldMatch(first = "minQty", second = "maxQty",operator = CrossFieldOperator.LE ,message = "最小數量必須小於等於最大數量")
})
public class ProductPriceQtyRange implements Serializable{
    /**
     * 最小數量
     */
    @Min(value = 0,message = "最小數量不合法")
    private int minQty;
    /**
     * 最大數量
     */
    @Min(value = 0,message = "最大數量不合法")
    private int maxQty;

    public int getMinQty() {
        return minQty;
    }

    public void setMinQty(int minQty) {
        this.minQty = minQty;
    }

    public int getMaxQty() {
        return maxQty;
    }

    public void setMaxQty(int maxQty) {
        this.maxQty = maxQty;
    }
}

 


需要在mvc的配置文件中增加如下節點以啟動檢驗

 <mvc:annotation-driven validator="validator">

  <bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean">
        <property name="providerClass" value="org.hibernate.validator.HibernateValidator"/>
        <property name="validationMessageSource" ref="messageSource"/>
    </bean>

    <bean id="messageSource" class="org.springframework.context.support.ReloadableResourceBundleMessageSource">
        <property name="useCodeAsDefaultMessage" value="false"/>  
        <property name="defaultEncoding" value="UTF-8"/>
    </bean>

經過上面在對象屬性上的數據檢驗註解,我們將大部分的數據檢驗邏輯從業務邏輯中轉移出去,不光是精簡了代碼還使得原本複雜的代碼變得簡單清晰,代碼的重覆利用率也增強了。

 

本文引用:

http://stackoverflow.com/questions/1972933/cross-field-validation-with-hibernate-validator-jsr-303

 




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

-Advertisement-
Play Games
更多相關文章
  • PHP(personal homepage)是HTML內嵌式語言,是一種在伺服器端執行的嵌入HTML文檔的腳本語言。由zend公司維護。為什麼要安裝web伺服器?因為我們瀏覽器要取數據,從web伺服器湖獲取。httpwatch工具可以獲取發送和接受的數據,有利於我們瞭解的更透徹。伺服器除Appche ...
  • 數據校驗指對數據合法性進行檢查,根據驗證數據的位置可以分為客戶端驗證和伺服器端驗證,今天隨筆主要寫的是實現伺服器端的數據驗證,伺服器端數據驗證主要特點: ·數據提交後在伺服器端驗證 ·防止繞過客戶端驗證提交的非法數據 ·可以在伺服器端處理數據前保證數據的合法性 Struts2中有兩種實現伺服器端驗證 ...
  • ...
  • 三觀是什麼鬼 當我們在討論「三觀一致」的時候是在討論些什麼? 我認為這個世界上本沒有「三觀」這一說法,說的人多了,也就有了「三觀」這個詞,當我們討論「三觀一致」其實並不是真的在說世界觀、價值觀、人生觀,而是再說,你們能不能玩到一起,吃到一起,睡到一起。就這麼簡單 官網下載 因為本文是基於 Windo ...
  • 1)Description: 1)Description: N! (N的階乘) 是非常大的數,計算公式為:N! = N * (N - 1) * (N - 2) * ... * 2 * 1)。現在需要知道N!有多少(十進位)位。 input:每行輸入1個正整數N。0 < N < 1000000 out ...
  • 1)名稱:memset()函數 2)別稱:char型初始化函數 3)功能: 將s所指向的某一塊記憶體中的每個位元組的內容全部設置為ch指定的ASCII值,塊的大小由第三個參數指定,這個函數通常為新申請的記憶體做初始化工作 4)用法: void *memset(void *s, char ch, unsig ...
  • Calculate the sum of two integers a and b, but you are not allowed to use the operator + and -. Example:Given a = 1 and b = 2, return 3. 個人思路:繞開+、-,利用 ...
  • 一、標識符 二、中置操作符 中置表達式,操作符位於兩個參數之間 1 to 10 1.to(10) 1 -> 10 1.->(10) 三、一元操作符 a.標識符() 1 toString 1.toString() +、-、!、~ 可以作為前置操作符,轉換成名為 unary_操作符 的方法調用 -a 和 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...