Springboot入門教程(9)-用validation做參數(shù)校驗及全局異常處理

參數(shù)校驗和異常處理也是后臺代碼中很重要的一部分继阻,如果每次都自己寫代碼做校驗就會很繁瑣,所以spring框架中也提供了validation組件來直接做參數(shù)校驗挪凑,本文就是講述validation組件的一些常見的用法,以及順便講一下如何全局的處理異常。

  1. 首先依然是先要在build.gradle的dependencies中添加依賴包
implementation "org.springframework.boot:spring-boot-starter-validation"
  1. 接著只要直接在java bean中配置參數(shù)條件就可以了纺酸,例如我們給teacher的幾個屬性加一些條件
    @Size(max=4, min=2, message="老師姓名應(yīng)為2-4字")
    private String name;
    @NotNull(message="老師性別不能為空")
    @Min(value = 0, message = "性別值只能為0或1,0:女址否,1:男")
    @Max(value = 1, message = "性別值只能為0或1餐蔬,0:女碎紊,1:男")
    private Integer gender;
    @NotNull(message="老師年齡不能為空")
    @Min(value = 20, message = "老師年齡不能小于20")
    @Max(value = 70, message = "老師年齡不能大于70")

這里的注解約束還有很多其他的,具體可以參考SpringBoot使用Validation校驗參數(shù)中的說明樊诺。也可以查看Hibernate Validator的官方文檔仗考,里面有更詳細(xì)的說明,還有一些不是很常見的特殊的注解約束词爬。

  1. 然后就可以在controller接口的參數(shù)前加上需要校驗的注解秃嗜,注解有兩種,一個是@Valid顿膨,一個是@Validated锅锨,這兩個大部分情況使用是一樣的。例如恋沃,這樣加上:
    @PostMapping(value = "/addTeacher", consumes = { "application/x-www-form-urlencoded" })
    @ResponseBody
    public ResponseData addTeacher(@Validated Teacher teacher)
    {
        if(teacher.getFile() != null){
            String fileName = FileUtil.upload(teacher.getFile(), path, teacher.getFile().getOriginalFilename());
            if ( fileName!= null){
                teacher.setImageUrl(fileName);
            }
        }
        teacherMapper.insertTeacher(teacher);
        ResponseData responseData = ResponseData.ok();
        return responseData;
    }

然后我們來運(yùn)行試試


傳入錯誤的參數(shù)

返回結(jié)果

可以看到返回了默認(rèn)格式的錯誤信息的json字符串必搞。
但是由于這個信息格式是默認(rèn)的,和我們自己定義的不一樣囊咏,前端可能就無法辨認(rèn)恕洲,這時就有兩個方法可以處理:
第一個方法是使用BindingResult,我們可以用BindingResult接收驗證的結(jié)果梅割,如果錯誤霜第,再按我們自己定義的格式返回錯誤信息。

@PostMapping(value = "/addTeacher", consumes = { "application/x-www-form-urlencoded" })
    @ResponseBody
    public ResponseData addTeacher(@Validated Teacher teacher, BindingResult bindingResult)
    {
        if (bindingResult.hasErrors()) {
            ResponseData responseData = new ResponseData(400, bindingResult.getFieldError().getDefaultMessage());
            return responseData;
        }
        if(teacher.getFile() != null){
            String fileName = FileUtil.upload(teacher.getFile(), path, teacher.getFile().getOriginalFilename());
            if ( fileName!= null){
                teacher.setImageUrl(fileName);
            }
        }
        teacherMapper.insertTeacher(teacher);
        ResponseData responseData = ResponseData.ok();
        return responseData;
    }

結(jié)果就會變成這樣


返回自定義的格式結(jié)果

第二個方法則就要引入本文的第二個課題了炮捧,就是全局的處理異常庶诡。因為如果每個校驗的異常都要這樣寫的話,那也是非常麻煩了咆课。所以Spring也提供了非常方便的全局異常的注解末誓,就是@RestControllerAdvice和@ExceptionHandler。我們就可以構(gòu)建如下的全局異常處理的類:

@RestControllerAdvice
public class GlobalExceptionHandler {
    /**
     * 處理Validated校驗異常
     * <p>
     * 注: 常見的ConstraintViolationException異常书蚪, 也屬于ValidationException異常
     *
     * @param e
     *         捕獲到的異常
     * @return 返回給前端的data
     */
    @ResponseStatus(code = HttpStatus.BAD_REQUEST)
    @ExceptionHandler(value = {BindException.class, ValidationException.class, MethodArgumentNotValidException.class})
    public ResponseData handleParameterVerificationException(Exception e) {
        String msg = null;
        /// BindException
        if (e instanceof BindException) {
            // getFieldError獲取的是第一個不合法的參數(shù)(P.S.如果有多個參數(shù)不合法的話)
            FieldError fieldError = ((BindException) e).getFieldError();
            if (fieldError != null) {
                msg = fieldError.getDefaultMessage();
            }
            /// MethodArgumentNotValidException
        } else if (e instanceof MethodArgumentNotValidException) {
            BindingResult bindingResult = ((MethodArgumentNotValidException) e).getBindingResult();
            // getFieldError獲取的是第一個不合法的參數(shù)(P.S.如果有多個參數(shù)不合法的話)
            FieldError fieldError = bindingResult.getFieldError();
            if (fieldError != null) {
                msg = fieldError.getDefaultMessage();
            }
            /// ValidationException 的子類異常ConstraintViolationException
        } else if (e instanceof ConstraintViolationException) {
            /*
             * ConstraintViolationException的e.getMessage()形如
             *     {方法名}.{參數(shù)名}: {message}
             *  這里只需要取后面的message即可
             */
            msg = e.getMessage();
            if (msg != null) {
                int lastIndex = msg.lastIndexOf(':');
                if (lastIndex >= 0) {
                    msg = msg.substring(lastIndex + 1).trim();
                }
            }
            /// ValidationException 的其它子類異常
        } else {
            msg = "處理參數(shù)時異常";
        }

        ResponseData responseData = new ResponseData(400, msg);
        return responseData;
    }

    @ExceptionHandler
    @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
    public ResponseData handleException(Exception ex){
        if (ex instanceof DataIntegrityViolationException) { // 數(shù)據(jù)庫操作異常
            if(ex.toString().contains("a foreign key constraint fails")){ //外鍵關(guān)聯(lián)問題喇澡,具體前端可以根據(jù)發(fā)送的請求判斷
                return new ResponseData(5001, "a foreign key constraint fails");
            }
        }
        return new ResponseData(500, "Internal Server Error");
    }
}

這里我寫了兩個方法,一個是專門處理參數(shù)校驗異常的殊校,我把它定義為BadRequest一類的返回晴玖,我主要考慮了三種參數(shù)異常的捕獲,例如前面的結(jié)果为流,就是BindException呕屎,在未加BindResult的處理之前,我們可以從控制臺打印的日志看出敬察。


控制臺打出的BindException異常

其實這個就是當(dāng)參數(shù)為Java bean且傳參方式為RequestParam就是直接在url地址上傳參的方式時校驗會返回的異常秀睛。
而第二種MethodArgumentNotValidException則是同樣參數(shù)為Java bean但是傳參方式為@RequestBody且applicationType為application/json的時候校驗會返回的異常,我們可以試一下


控制臺打出的MethodArgumentNotValidException異常

而第三種ConstraintViolationException呢莲祸,則是參數(shù)為普通類型的直接在參數(shù)前加校驗條件的異常返回類型蹂安。例如在findSubjects的參數(shù)上加上校驗
@GetMapping(value = "/subjects")
    public ResponseDataNew<ListWithPageData<Subject>> findSubjects(final String name, @Min(value = 0, message = "頁碼不能小于0")final Integer index, final Integer size){
        Page<Subject> page = PageHelper.startPage(index + 1, size);
        List<Subject> subjectList = subjectMapper.findSubjects(name);
        ResponseDataNew<ListWithPageData<Subject>> response = new ResponseDataNew<>();
        response.ok();
        ListWithPageData<Subject> data = new ListWithPageData<>();
        data.setPageCount(page.getPages());
        data.setTotal(page.getTotal());
        data.setList(subjectList);
        response.setData(data);
        return response;
    }

這里還涉及到@Validated的另一個用法椭迎,就是加在類上的注解,只有這樣

@RestController
@RequestMapping(value = "/subject")
@Validated
public class SubjectController {
...
}

直接加在參數(shù)前的校驗注解才會有用田盈。傳入index為-1時就會拋出這個異常


控制臺打出的ConstraintViolationException異常

另外我還寫了一個方法用來捕獲其他類型的異常畜号,比如這個外鍵關(guān)聯(lián)的異常。@ExceptionHandler這個注解就是可以指定捕獲的Exception的類型允瞧,如果沒有指定简软,那么就會捕獲任意類型。

此外瓷式,@Validated還支持分組替饿,比如當(dāng)我們新建一條數(shù)據(jù)時,id是必然為空的贸典,而更新數(shù)據(jù)時视卢,id又必須不為空,這時就可以用到這個廊驼。
(1)首先据过,我們在entity包中分別建兩個接口Insert和Update

public interface Insert extends Default {
}
public interface Update extends Default {
}

(2)接著,以Subject為例妒挎,需要在id上加兩組注解

    @Schema(example = "1")
    @Null(groups = {Insert.class})
    @NotNull(groups = {Update.class}, message="id不能為空")
    private Long id;

(3)分別在新增和更新的接口上加上對應(yīng)的注解绳锅,如下

    @PostMapping(value = "/addSubject", consumes = { "application/x-www-form-urlencoded" })
    public ResponseData addSubject(@Validated(value = Insert.class) Subject subject){
        subjectMapper.insertSubject(subject.getName());
        ResponseData responseData = ResponseData.ok();
        return responseData;
    }

    @PostMapping(value = "/editSubject", consumes = { "application/x-www-form-urlencoded" })
    public ResponseData editSubject(@Validated(value = Update.class) Subject subject)
    {
        subjectMapper.updateSubject(subject.getId(), subject.getName());
        ResponseData responseData = ResponseData.ok();
        return responseData;
    }

但是這樣加完會有個問題,就是在swagger上酝掩,我們會發(fā)現(xiàn)addSubject的接口id的參數(shù)仍然是required的鳞芙,這似乎是一個bug。而用Postman測試期虾,結(jié)果則是正常的


addSubject校驗

editSubject校驗

不過原朝,我也試了一下,如果把傳參方式改為@RequestBody就是application/json的話也可以解決這個問題镶苞。
需要注意的是@Valid是不支持這樣分組的喳坠,這是這兩個注解其中一個差異。

@Validated和@Valid還有一個差異在于@Valid支持嵌套校驗茂蚓、而@Validated不支持壕鹉。這是什么意思呢?比如我需要做一個批量新增的功能聋涨,所以我傳參的時候會傳一個list晾浴,就像這樣

@PostMapping(value = "/addSubjects")
    public ResponseData addSubjects(@Validated(value = Insert.class) @RequestBody List<Subject> subjects){
        subjectMapper.insertSubjects(subjects);
        ResponseData responseData = ResponseData.ok();
        return responseData;
    }

但是這時候我們加的這個@Validated的注解會發(fā)現(xiàn)是不起作用的,就是因為它不支持嵌套牍白,而要驗證的對象包在List中脊凰。這時我們只能把它改為@Valid,分組也就沒辦法使用了淹朋。還要注意的是同樣要在SubjectController類上加了@Validated注解才有用笙各。
不過也還有一種方法可以同時解決這兩個問題,就是自定義實現(xiàn)一個List ValidatedList础芍,這個方法的話可以參考使用@Validated校驗List接口參數(shù)的兩種方式這篇博客杈抢。

最后再說一下的是,validation還支持自定義的校驗仑性,這個也可以參考SpringBoot使用Validation校驗參數(shù)這篇博客惶楼,我這里也就不再詳細(xì)說明了。
代碼依舊可以參考我在github上面的代碼https://github.com/ahuadoreen/studentmanager诊杆。

參考文檔
SpringBoot使用Validation校驗參數(shù)
Spring 參數(shù)校驗的異常處理
使用@Validated校驗List接口參數(shù)的兩種方式

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末歼捐,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子晨汹,更是在濱河造成了極大的恐慌豹储,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,248評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件淘这,死亡現(xiàn)場離奇詭異剥扣,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)铝穷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評論 2 381
  • 文/潘曉璐 我一進(jìn)店門钠怯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人曙聂,你說我怎么就攤上這事晦炊。” “怎么了宁脊?”我有些...
    開封第一講書人閱讀 153,443評論 0 344
  • 文/不壞的土叔 我叫張陵断国,是天一觀的道長。 經(jīng)常有香客問我朦佩,道長并思,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,475評論 1 279
  • 正文 為了忘掉前任语稠,我火速辦了婚禮宋彼,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘仙畦。我一直安慰自己输涕,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,458評論 5 374
  • 文/花漫 我一把揭開白布慨畸。 她就那樣靜靜地躺著莱坎,像睡著了一般。 火紅的嫁衣襯著肌膚如雪寸士。 梳的紋絲不亂的頭發(fā)上檐什,一...
    開封第一講書人閱讀 49,185評論 1 284
  • 那天碴卧,我揣著相機(jī)與錄音,去河邊找鬼乃正。 笑死住册,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 38,451評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼毫捣,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了叹阔?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,112評論 0 261
  • 序言:老撾萬榮一對情侶失蹤传睹,失蹤者是張志新(化名)和其女友劉穎耳幢,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蒋歌,經(jīng)...
    沈念sama閱讀 43,609評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡帅掘,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,083評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了堂油。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片修档。...
    茶點故事閱讀 38,163評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖府框,靈堂內(nèi)的尸體忽然破棺而出吱窝,到底是詐尸還是另有隱情,我是刑警寧澤迫靖,帶...
    沈念sama閱讀 33,803評論 4 323
  • 正文 年R本政府宣布院峡,位于F島的核電站,受9級特大地震影響系宜,放射性物質(zhì)發(fā)生泄漏照激。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,357評論 3 307
  • 文/蒙蒙 一盹牧、第九天 我趴在偏房一處隱蔽的房頂上張望俩垃。 院中可真熱鬧,春花似錦汰寓、人聲如沸口柳。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽跃闹。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間望艺,已是汗流浹背苛秕。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留找默,地道東北人想帅。 一個月前我還...
    沈念sama閱讀 45,636評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像啡莉,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子旨剥,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,925評論 2 344

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