Spring Boot 異常處理,值得學(xué)習(xí)!
背景:
個(gè)人也是一邊學(xué)習(xí)一邊做,難免有疏漏的地方,希望朋友們?cè)诓粚?duì)的地方提醒下。
技術(shù)棧:
- springBoot 2.5.3
- Mybatis-plus
- thymeleaf
- mysql
該博客簡(jiǎn)單記錄一下自己的學(xué)習(xí)過(guò)程,如果后面作出簡(jiǎn)單的demo來(lái),那么會(huì)整理其中的細(xì)節(jié)發(fā)布出來(lái)。
推薦一個(gè) Spring Boot 基礎(chǔ)教程及實(shí)戰(zhàn)示例:
??https://github.com/javastacks/spring-boot-best-practice ??
異常處理
個(gè)人粗糙的學(xué)習(xí)和理解,我把異常錯(cuò)誤分為三個(gè),4xx,5xx,error(自定義)
常用的是404和500響應(yīng)
- 404 (未找到) 服務(wù)器找不到請(qǐng)求的網(wǎng)頁(yè)
- 服務(wù)器內(nèi)部錯(cuò)誤 服務(wù)器遇到錯(cuò)誤,無(wú)法完成請(qǐng)求
在templates文件下創(chuàng)建404.html、500.html和error.html
如果創(chuàng)建的是4xx.html和5xx.html,那么當(dāng)頁(yè)面找不到或者程序內(nèi)部錯(cuò)誤的話(huà),SpringBoot自動(dòng)匹配到這兩個(gè)頁(yè)面,具體的源碼在這里就不分析,感興趣的可以去學(xué)習(xí)下。
“
當(dāng)我想將錯(cuò)誤信息返回到我自定義的頁(yè)面怎么寫(xiě)?
”
我們自頂向下來(lái)思考:
- 我們?cè)L問(wèn)頁(yè)面的時(shí)候,頁(yè)面發(fā)生錯(cuò)誤也好,找不到也好,是不是都要走它請(qǐng)求的Url,那我們?cè)趺刺幚磉@個(gè)請(qǐng)求呢?
- 這時(shí)候我們自然而然的想到攔截器,所以編寫(xiě)ControllerExceptionHandler類(lèi),來(lái)專(zhuān)門(mén)攔截所有的異常請(qǐng)求。
- 當(dāng)我們處理完異常后,把這個(gè)流放行,或者返回我們需要的自定義頁(yè)面上。
這樣就實(shí)現(xiàn)了我們上述的需求。
所需要的技術(shù)點(diǎn):
- @ControllerAdvice
- 配合 @ExceptionHandler注解結(jié)合使用,當(dāng)異常拋到controller層時(shí),可以對(duì)異常進(jìn)行統(tǒng)一的處理,規(guī)定返回的json格式或者跳轉(zhuǎn)到指定的錯(cuò)誤頁(yè)面等.
- @ExceptionHandler(Exception.class) // 表示 捕獲 全部異常
- ModelAndView 其實(shí)就是兩個(gè)作用,一個(gè)是指定返回頁(yè)面,另一個(gè)是在返回頁(yè)面的同時(shí)添加屬性
通過(guò)Logger來(lái)打印獲取相關(guān)的異常信息:
//獲取異常的信息
logger.error(() -> {
return String.format("Request URL : %s,Exception : %s ", request.getRequestURL(),e);
},e);
在這里自己走了一些彎路,小伙子自己走窄了,導(dǎo)包倒錯(cuò)了??。最新面試題整理好了,大家可以在Java面試庫(kù)小程序在線(xiàn)刷題。
這里我導(dǎo)的包是mybatis中的:
import org.mybatis.logging.Logger;
import org.mybatis.logging.LoggerFactory;
傳參一直不對(duì),只能走源碼的路子了,看看自己調(diào)用的什么玩意:最后就寫(xiě)成了上述的代碼格式。
我看其他人寫(xiě)的代碼的時(shí)候才恍然大明白,如下:
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
logger.error("Requst URL : {},Exception : {}", request.getRequestURL(),e);
可能有的朋友感覺(jué)沒(méi)啥,但是我自己調(diào)了半個(gè)小時(shí)才出來(lái),菜是原罪。
Logger這個(gè)只是在控制臺(tái)輸出,或者綁定了日志,會(huì)輸出到日志中。
接著上述問(wèn)題通過(guò)ModelAndView類(lèi)實(shí)現(xiàn):
“
ModelAndView像極了Model。
”
敲黑板:來(lái)新需求了!
如果我想從所有的異常中剝離出404異常并返回給SpringBoot讓他自動(dòng)處理頁(yè)面怎么解。最新 Spring Boot 面試題整理好了,大家可以在Java面試庫(kù)小程序在線(xiàn)刷題。
按照學(xué)習(xí)的來(lái)說(shuō)。當(dāng)頁(yè)面為null的時(shí)候,向上拋出一個(gè)自定義的異常類(lèi),該異常類(lèi)標(biāo)注了異常狀態(tài)。
throw new NotFoundException("博客找不到,請(qǐng)聯(lián)系管理員"); //自定義類(lèi)
HttpStatus.NOT_FOUND)(
public class NotFoundException extends RuntimeException{
public NotFoundException() {
super();
}
public NotFoundException(String message) {
super(message);
}
}
然后我們可以在總的異常中通過(guò)AnnotationUtils.findAnnotation通過(guò)傳入AnnotatedElement和注解類(lèi)型來(lái)查找方法或者類(lèi)對(duì)象上的注解。
如果滿(mǎn)足條件可以?huà)伋霎惓W孲pringBoot接管。
if (AnnotationUtils.findAnnotation(e.getClass(), ResponseStatus.class) != null) {
throw e;
}