SpringBoot 實(shí)戰(zhàn) (十四) | 統(tǒng)一處理異常

微信公眾號(hào):一個(gè)優(yōu)秀的廢人
如有問題或建議,請(qǐng)后臺(tái)留言殉农,我會(huì)盡力解決你的問題肛炮。

前言

如題惊橱,今天介紹 SpringBoot 是如何統(tǒng)一處理全局異常的蚪腐。SpringBoot 中的全局異常處理主要起作用的兩個(gè)注解是 @ControllerAdvice@ExceptionHandler ,其中 @ControllerAdvice 是組件注解李皇,添加了這個(gè)注解的類能夠攔截 Controller 的請(qǐng)求,而 ExceptionHandler 注解可以設(shè)置全局處理控制里的異常類型來攔截要處理的異常宙枷。 比如:@ExceptionHandler(value = NullPointException.class) 掉房。

準(zhǔn)備工作

  • SpringBoot 2.1.3
  • IDEA
  • JDK 8

依賴配置

<dependencies>
        <!-- JPA 依賴 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>
        <!-- web 依賴 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <!-- mysql 連接類 -->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>
        <!-- lombok 依賴 -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
        <!-- 單元測(cè)試依賴 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

配置文件

spring:
  # 數(shù)據(jù)庫(kù)相關(guān)
  datasource:
    driver-class-name: com.mysql.jdbc.Driver
    url: jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&characterEncoding=utf8&serverTimezone=UTC&useSSL=true
    username: root
    password: 123456

  jpa:
    hibernate:
      ddl-auto: update   #ddl-auto:設(shè)為 create 表示每次都重新建表
    show-sql: true

返回的消息類

public class Message<T> implements Serializable {

    /**
     * 狀態(tài)碼
     */
    private Integer code;

    /**
     * 返回信息
     */
    private String message;

    /**
     * 返回的數(shù)據(jù)類
     */
    private T data;

    /**
     * 時(shí)間
     */
    private Long time;

    // getter、setter 以及 構(gòu)造方法略慰丛。卓囚。。
}

工具類

用于處理返回的數(shù)據(jù)以及信息類诅病,代碼注釋很詳細(xì)不說了哪亿。

public class MessageUtil {

    /**
     * 成功并返回?cái)?shù)據(jù)實(shí)體類
     * @param o
     * @param <E>
     * @return
     */
    public static <E>Message<E> ok(E o){
        return new Message<>(200, "success", o, new Date().getTime());
    }

    /**
     * 成功粥烁,但無(wú)數(shù)據(jù)實(shí)體類返回
     * @return
     */
    public static <E>Message<E> ok(){
        return new Message<>(200, "success", null, new Date().getTime());
    }

    /**
     * 失敗,有自定義異常返回
     * @param code
     * @param msg
     * @return
     */
    public static <E>Message<E> error(Integer code,String msg){
        return new Message<>(code, msg, null, new Date().getTime());
    }
}

自定義異常

通過繼承 RuntimeException 蝇棉,聲明 code 用于定義不同類型的自定義異常讨阻。主要是用于異常攔截出獲取 code 并將 code 設(shè)置到消息類中返回。

public class CustomException extends RuntimeException{

    /**
     * 狀態(tài)碼
     */
    private Integer code;

    public Integer getCode() {
        return code;
    }

    public void setCode(Integer code) {
        this.code = code;
    }

    public CustomException(Integer code, String message){

        super(message);
        this.code = code;

    }
}

異常攔截類

通過加入 @RestControllerAdvice 來聲明該類可攔截 Controller 請(qǐng)求篡殷,同時(shí)在 handle方法加入 @ExceptionHandler 并在該注解中指定要攔截的異常類钝吮。

@RestControllerAdvice // 控制器增強(qiáng)處理(返回 JSON 格式數(shù)據(jù)),添加了這個(gè)注解的類能被 classpath 掃描自動(dòng)發(fā)現(xiàn)
public class ExceptionHandle {

    @ExceptionHandler(value = Exception.class) // 捕獲 Controller 中拋出的指定類型的異常板辽,也可以指定其他異常
    public <E>Message<E> handler(Exception exception){

        if (exception instanceof CustomException){
            CustomException customException = (CustomException) exception;
            return MessageUtil.error(customException.getCode(), customException.getMessage());
        } else {
            return MessageUtil.error(120, "異常信息:" + exception.getMessage());
        }
    }
}

這里只對(duì)自定義異常以及未知異常進(jìn)行處理奇瘦,如果你在某方法中明確知道可能會(huì)拋出某個(gè)異常,可以加多一個(gè)特定的處理劲弦。比如說你明確知道該方法可能拋出 NullPointException 可以追加 NullPointException 的處理:

if (exception instanceof CustomException){
     CustomException customException = (CustomException) exception;
     return MessageUtil.error(customException.getCode(), customException.getMessage());
} else if (exception instanceof NullPointException ){
     return MessageUtil.error(500, "空指針異常信耳标!");
} else {
     return MessageUtil.error(120, "異常信息:" + exception.getMessage());
}

controller 層

@RestController
@RequestMapping("/student")
public class StudentController {

    @Autowired
    private StudentService studentService;

    @GetMapping("/{id}")
    public Message<Student> findStudentById(@PathVariable("id") Integer id){

        if (id < 0){
            //測(cè)試自定義錯(cuò)誤
            throw new CustomException(110, "參數(shù)不能是負(fù)數(shù)!");

        } else if (id == 0){
            //硬編碼邑跪,為了測(cè)試
            Integer i = 1/id;
            return null;
        } else {
            Student student = studentService.findStudentById(id);
            return MessageUtil.ok(student);
        }
    }
}

完整代碼

https://github.com/turoDog/Demo/tree/master/springboot_exception_demo

如果覺得對(duì)你有幫助次坡,請(qǐng)給個(gè) Star 再走唄,非常感謝呀袱。

Postman 測(cè)試

訪問 http://localhost:8080/student/5 測(cè)試正常返回?cái)?shù)據(jù)結(jié)果贸毕。

返回正常結(jié)果

訪問 http://localhost:8080/student/0 測(cè)試未知異常的結(jié)果。

測(cè)試未知異常的情況

訪問 http://localhost:8080/student/-11 測(cè)試自定義異常的結(jié)果夜赵。

測(cè)試自定義異常情況

后語(yǔ)

如果本文對(duì)你哪怕有一丁點(diǎn)幫助明棍,請(qǐng)幫忙點(diǎn)好看。你的好看是我堅(jiān)持寫作的動(dòng)力寇僧。

另外摊腋,關(guān)注之后在發(fā)送 1024 可領(lǐng)取免費(fèi)學(xué)習(xí)資料。資料內(nèi)容詳情請(qǐng)看這篇舊文:Python嘁傀、C++兴蒸、Java、Linux细办、Go橙凳、前端、算法資料分享

一個(gè)優(yōu)秀的廢人
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末笑撞,一起剝皮案震驚了整個(gè)濱河市岛啸,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌茴肥,老刑警劉巖坚踩,帶你破解...
    沈念sama閱讀 207,113評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異瓤狐,居然都是意外死亡瞬铸,警方通過查閱死者的電腦和手機(jī)批幌,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來嗓节,“玉大人荧缘,你說我怎么就攤上這事∩庹” “怎么了胜宇?”我有些...
    開封第一講書人閱讀 153,340評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)恢着。 經(jīng)常有香客問我桐愉,道長(zhǎng),這世上最難降的妖魔是什么掰派? 我笑而不...
    開封第一講書人閱讀 55,449評(píng)論 1 279
  • 正文 為了忘掉前任从诲,我火速辦了婚禮,結(jié)果婚禮上靡羡,老公的妹妹穿的比我還像新娘系洛。我一直安慰自己,他們只是感情好略步,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評(píng)論 5 374
  • 文/花漫 我一把揭開白布描扯。 她就那樣靜靜地躺著,像睡著了一般趟薄。 火紅的嫁衣襯著肌膚如雪绽诚。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,166評(píng)論 1 284
  • 那天杭煎,我揣著相機(jī)與錄音恩够,去河邊找鬼。 笑死羡铲,一個(gè)胖子當(dāng)著我的面吹牛蜂桶,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播也切,決...
    沈念sama閱讀 38,442評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼扑媚,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了雷恃?” 一聲冷哼從身側(cè)響起疆股,我...
    開封第一講書人閱讀 37,105評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎褂萧,沒想到半個(gè)月后押桃,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體葵萎,經(jīng)...
    沈念sama閱讀 43,601評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡导犹,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評(píng)論 2 325
  • 正文 我和宋清朗相戀三年唱凯,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片谎痢。...
    茶點(diǎn)故事閱讀 38,161評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡磕昼,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出节猿,到底是詐尸還是另有隱情票从,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評(píng)論 4 323
  • 正文 年R本政府宣布滨嘱,位于F島的核電站峰鄙,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏太雨。R本人自食惡果不足惜吟榴,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望囊扳。 院中可真熱鬧吩翻,春花似錦、人聲如沸锥咸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)搏予。三九已至熊锭,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間缔刹,已是汗流浹背球涛。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評(píng)論 1 261
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留校镐,地道東北人亿扁。 一個(gè)月前我還...
    沈念sama閱讀 45,618評(píng)論 2 355
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像鸟廓,于是被迫代替她去往敵國(guó)和親从祝。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評(píng)論 2 344

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