Spring 校驗(yàn)器(Validator)

Spring校驗(yàn)器,參數(shù)校驗(yàn)從此簡(jiǎn)單腔稀。


image.png

應(yīng)用在執(zhí)行業(yè)務(wù)邏輯之前盆昙,必須通過(guò)校驗(yàn)保證接受到的輸入數(shù)據(jù)是合法正確的,但很多時(shí)候同樣的校驗(yàn)出現(xiàn)了多次焊虏,在不同的層弱左,不同的方法上,導(dǎo)致代碼冗余炕淮,浪費(fèi)時(shí)間拆火,違反DRY原則。

  1. 每一個(gè)控制器都要校驗(yàn)
  2. 過(guò)多的校驗(yàn)參數(shù)會(huì)導(dǎo)致代碼太長(zhǎng)
  3. 代碼的復(fù)用率太差,同樣的代碼如果出現(xiàn)多次们镜,在業(yè)務(wù)越來(lái)越復(fù)雜的情況下币叹,維護(hù)成本呈指數(shù)上升。

可以考慮把校驗(yàn)的代碼封裝起來(lái)模狭,來(lái)解決出現(xiàn)的這些問(wèn)題颈抚。

JSR-303

JSR-303是Java為Bean數(shù)據(jù)合法性校驗(yàn)提供的標(biāo)準(zhǔn)框架,它定義了一套可標(biāo)注在成員變量嚼鹉,屬性方法上的校驗(yàn)注解贩汉。
Hibernate Validation提供了這套標(biāo)準(zhǔn)的實(shí)現(xiàn),在我們引入Spring Boot web starter或者Spring boot starter validation的時(shí)候锚赤,默認(rèn)會(huì)引入Hibernate Validation匹舞。

用法實(shí)例

說(shuō)了這么多廢話,上代碼线脚。

  1. 引入SpringBoot項(xiàng)目
      <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.hibernate.validator</groupId>
            <artifactId>hibernate-validator</artifactId>
        </dependency>
       <!-- 引入lomhok --> 
       <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>        
        
  1. 編寫(xiě)校驗(yàn)對(duì)象
@Data
public class User {
    // 名字不允許為空赐稽,并且名字的長(zhǎng)度在2位到30位之間
    // 如果名字的長(zhǎng)度校驗(yàn)不通過(guò),那么提示錯(cuò)誤信息
    @NotNull
    @Size(min=2, max=30,message = "請(qǐng)檢查名字的長(zhǎng)度是否有問(wèn)題")
    private String name;

    // 不允許為空浑侥,并且年齡的最小值為18
    @NotNull
    @Min(18)
    private Integer age;
}

  1. 創(chuàng)建控制器

@SpringBootApplication
@RestController
public class UserApplication{
    public static void main(String[] args) {
        SpringApplication.run(UserApplication.class,args);
    }
    
    // 1. 要校驗(yàn)的參數(shù)前姊舵,加上@Valid注解
    // 2. 緊隨其后的,跟上一個(gè)BindingResult來(lái)存儲(chǔ)校驗(yàn)信息
    @RequestMapping("/test1")
    public Object test1(
            @Valid User user,
            BindingResult bindingResult
    ) {
        //如果檢驗(yàn)出了問(wèn)題寓落,就返回錯(cuò)誤信息
        // 這里我們返回的是全部的錯(cuò)誤信息括丁,實(shí)際中可根據(jù)bindingResult的方法根據(jù)需要返回自定義的信息。
        // 通常的解決方案為:JSR-303 + 全局ExceptionHandler
        if (bindingResult.hasErrors()){
            return bindingResult.getAllErrors();
        }
        return "OK";
    }
    
}

  1. 運(yùn)行應(yīng)用

稍作演示下運(yùn)行的結(jié)果伶选,可以看出校驗(yàn)框架已經(jīng)生效了躏将。


校驗(yàn)?zāi)挲g
校驗(yàn)名稱(chēng)

校驗(yàn)通過(guò)

常見(jiàn)的校驗(yàn)注解

@Null 被注釋的元素必須為 null
@NotNull 被注釋的元素必須不為 null
@AssertTrue 被注釋的元素必須為 true
@AssertFalse 被注釋的元素必須為 false
@Min(value) 被注釋的元素必須是一個(gè)數(shù)字,其值必須大于等于指定的最小值
@Max(value) 被注釋的元素必須是一個(gè)數(shù)字考蕾,其值必須小于等于指定的最大值
@DecimalMin(value) 被注釋的元素必須是一個(gè)數(shù)字,其值必須大于等于指定的最小值
@DecimalMax(value) 被注釋的元素必須是一個(gè)數(shù)字会宪,其值必須小于等于指定的最大值
@Size(max=, min=) 被注釋的元素的大小必須在指定的范圍內(nèi)
@Digits (integer, fraction) 被注釋的元素必須是一個(gè)數(shù)字肖卧,其值必須在可接受的范圍內(nèi)
@Past 被注釋的元素必須是一個(gè)過(guò)去的日期
@Future 被注釋的元素必須是一個(gè)將來(lái)的日期
@Pattern(regex=,flag=) 被注釋的元素必須符合指定的正則表達(dá)式

Hibernate Validator提供的校驗(yàn)注解:
@NotBlank(message =) 驗(yàn)證字符串非null,且長(zhǎng)度必須大于0
@Email 被注釋的元素必須是電子郵箱地址
@Length(min=,max=) 被注釋的字符串的大小必須在指定的范圍內(nèi)
@NotEmpty 被注釋的字符串的必須非空
@Range(min=,max=,message=) 被注釋的元素必須在合適的范圍內(nèi)

自定義校驗(yàn)注解

有時(shí)候掸鹅,第三方庫(kù)中并沒(méi)有我們想要的校驗(yàn)類(lèi)型塞帐,好在系統(tǒng)提供了很好的擴(kuò)展能力巍沙,我們可以自定義檢驗(yàn)葵姥。
比如句携,我們想校驗(yàn)用戶(hù)的手機(jī)格式榔幸,寫(xiě)手機(jī)號(hào)碼校驗(yàn)器

  1. 編寫(xiě)校驗(yàn)注解
// 我們可以直接拷貝系統(tǒng)內(nèi)的注解如@Min,復(fù)制到我們新的注解中,然后根據(jù)需要修改鳞陨。
@Target({METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER})
@Retention(RUNTIME)
@Documented
//注解的實(shí)現(xiàn)類(lèi)灰嫉。
@Constraint(validatedBy = {IsMobileValidator.class})
public @interface IsMobile {
    //校驗(yàn)錯(cuò)誤的默認(rèn)信息
    String message() default "手機(jī)號(hào)碼格式有問(wèn)題";

    //是否強(qiáng)制校驗(yàn)
    boolean isRequired() default false;
    
    Class<?>[] groups() default {};

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

  1. 編寫(xiě)具體的實(shí)現(xiàn)類(lèi)
    我們知道注解只是一個(gè)標(biāo)記奇适,真正的邏輯還要在特定的類(lèi)中實(shí)現(xiàn)号胚,上一步的注解指定了實(shí)現(xiàn)校驗(yàn)功能的類(lèi)為IsMobileValidator永高。
// 自定義注解一定要實(shí)現(xiàn)ConstraintValidator接口奧蔬胯,里面的兩個(gè)參數(shù)
// 第一個(gè)為 具體要校驗(yàn)的注解
// 第二個(gè)為 校驗(yàn)的參數(shù)類(lèi)型
public class IsMobileValidator implements ConstraintValidator<IsMobile, String> {

    private boolean required = false;

    private static final Pattern mobile_pattern = Pattern.compile("1\\d{10}");
    //工具方法资盅,判斷是否是手機(jī)號(hào)
    public static boolean isMobile(String src) {
        if (StringUtils.isEmpty(src)) {
            return false;
        }
        Matcher m = mobile_pattern.matcher(src);
        return m.matches();
    }

    @Override
    public void initialize(IsMobile constraintAnnotation) {
        required = constraintAnnotation.isRequired();
    }

    @Override
    public boolean isValid(String phone, ConstraintValidatorContext constraintValidatorContext) {
        //是否為手機(jī)號(hào)的實(shí)現(xiàn)
        if (required) {
            return isMobile(phone);
        } else {
            if (StringUtils.isEmpty(phone)) {
                return true;
            } else {
                return isMobile(phone);
            }
        }
    }
    
}
  1. 測(cè)試自定義注解的功能
@Data
public class User {
    @NotNull
    @Size(min=2, max=30,message = "請(qǐng)檢查名字的長(zhǎng)度是否有問(wèn)題")
    private String name;

    @NotNull
    @Min(18)
    private Integer age;

    //這里是新添加的注解奧
    @IsMobile
    private String phone;
}
  1. 測(cè)試


    通過(guò)
手機(jī)號(hào)有問(wèn)題

可以看出自定義的注解已經(jīng)生效了狈邑。
我們還可以繼續(xù)優(yōu)化的地方蹂匹,新建一個(gè)全局的異常碘菜,如果校驗(yàn)失敗的話,拋出全局的業(yè)務(wù)異常限寞,捕獲業(yè)務(wù)異常忍啸,然后返回用戶(hù)友好的提示信息。

額外

也可以通過(guò)方法的校驗(yàn)履植。

  1. 控制器上添加@Validated注解
  2. 在控制器的方法上添加校驗(yàn)注解计雌,@Min,@Max等玫霎。
@Validated
@RestController
@SpringBootApplication
public class UserApplication{
    public static void main(String[] args) {
        SpringApplication.run(UserApplication.class,args);
    }

    @RequestMapping("/test2")
    public String test2(
            @IsMobile String phone

    ){
        return phone + "ok";
    }

    @ExceptionHandler(ConstraintViolationException.class)
    @ResponseBody
    public Object handleConstraintViolationException(ConstraintViolationException cve){

        HashSet<String> messageSet = new HashSet();
        for (ConstraintViolation constraintViolation : cve.getConstraintViolations()) {
            messageSet.add(constraintViolation.getMessage());
        }
        return messageSet;
    }

}

類(lèi)的校驗(yàn)規(guī)則

最后

通過(guò)使用校驗(yàn)器凿滤,所有的控制器,我們都不用再去做校驗(yàn)啦庶近,代碼再回看是不是清爽很多翁脆。我們寫(xiě)代碼很簡(jiǎn)答,但是一定要想到如何把代碼寫(xiě)的更簡(jiǎn)單鼻种,更清晰反番,更利于維護(hù),寫(xiě)重復(fù)的代碼是在浪費(fèi)自己的時(shí)間奧叉钥。

以后再碰到參數(shù)校驗(yàn)的情況罢缸,首先想到的不是直接就去校驗(yàn),可以查找自己是否寫(xiě)過(guò)某一類(lèi)的驗(yàn)證器投队,可以直接拿來(lái)即用枫疆。

希望能幫助大家。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末敷鸦,一起剝皮案震驚了整個(gè)濱河市息楔,隨后出現(xiàn)的幾起案子寝贡,更是在濱河造成了極大的恐慌,老刑警劉巖钞螟,帶你破解...
    沈念sama閱讀 216,919評(píng)論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件兔甘,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡鳞滨,警方通過(guò)查閱死者的電腦和手機(jī)洞焙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,567評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)拯啦,“玉大人澡匪,你說(shuō)我怎么就攤上這事“矗” “怎么了唁情?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,316評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)甫匹。 經(jīng)常有香客問(wèn)我甸鸟,道長(zhǎng),這世上最難降的妖魔是什么兵迅? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,294評(píng)論 1 292
  • 正文 為了忘掉前任抢韭,我火速辦了婚禮,結(jié)果婚禮上恍箭,老公的妹妹穿的比我還像新娘刻恭。我一直安慰自己,他們只是感情好扯夭,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,318評(píng)論 6 390
  • 文/花漫 我一把揭開(kāi)白布鳍贾。 她就那樣靜靜地躺著,像睡著了一般交洗。 火紅的嫁衣襯著肌膚如雪骑科。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,245評(píng)論 1 299
  • 那天构拳,我揣著相機(jī)與錄音咆爽,去河邊找鬼。 笑死隐圾,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的掰茶。 我是一名探鬼主播暇藏,決...
    沈念sama閱讀 40,120評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼濒蒋!你這毒婦竟也來(lái)了盐碱?” 一聲冷哼從身側(cè)響起把兔,我...
    開(kāi)封第一講書(shū)人閱讀 38,964評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎瓮顽,沒(méi)想到半個(gè)月后县好,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,376評(píng)論 1 313
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡暖混,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,592評(píng)論 2 333
  • 正文 我和宋清朗相戀三年缕贡,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片拣播。...
    茶點(diǎn)故事閱讀 39,764評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡晾咪,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出贮配,到底是詐尸還是另有隱情谍倦,我是刑警寧澤,帶...
    沈念sama閱讀 35,460評(píng)論 5 344
  • 正文 年R本政府宣布泪勒,位于F島的核電站昼蛀,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏圆存。R本人自食惡果不足惜叼旋,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,070評(píng)論 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望辽剧。 院中可真熱鬧送淆,春花似錦、人聲如沸怕轿。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,697評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)撞羽。三九已至阐斜,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間诀紊,已是汗流浹背谒出。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,846評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留邻奠,地道東北人笤喳。 一個(gè)月前我還...
    沈念sama閱讀 47,819評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像碌宴,于是被迫代替她去往敵國(guó)和親杀狡。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,665評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理贰镣,服務(wù)發(fā)現(xiàn)呜象,斷路器膳凝,智...
    卡卡羅2017閱讀 134,654評(píng)論 18 139
  • 1.1 spring IoC容器和beans的簡(jiǎn)介 Spring 框架的最核心基礎(chǔ)的功能是IoC(控制反轉(zhuǎn))容器,...
    simoscode閱讀 6,713評(píng)論 2 22
  • Spring Web MVC Spring Web MVC 是包含在 Spring 框架中的 Web 框架恭陡,建立于...
    Hsinwong閱讀 22,394評(píng)論 1 92
  • 1.1 為什么要用動(dòng)態(tài)代理 學(xué)一個(gè)東西蹬音,感覺(jué)比較好的方式是問(wèn)自己它為什么要存在?學(xué)習(xí)java動(dòng)態(tài)代理首先需要...
    tracy_668閱讀 798評(píng)論 1 1
  • 文件讀取 讀取read.text內(nèi)容至file.html中 同步讀取 異步讀取
    余生筑閱讀 68評(píng)論 0 0