溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何在spring boot中實(shí)現(xiàn)默認(rèn)異常處理

發(fā)布時(shí)間:2021-05-27 17:52:30 來(lái)源:億速云 閱讀:234 作者:Leah 欄目:編程語(yǔ)言

這篇文章給大家介紹如何在spring boot中實(shí)現(xiàn)默認(rèn)異常處理,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助。

莫名的攔截器

項(xiàng)目配置了一個(gè)攔截器,專門用來(lái)對(duì)用戶進(jìn)行驗(yàn)證是否登錄的,這個(gè)是前提.在使用response.setStatus()方法時(shí),前臺(tái)能正確的接受到傳入的狀態(tài)碼,而使用response.sendError()時(shí),前臺(tái)卻接受到的一直是401用戶未登錄的狀態(tài)碼,打了斷點(diǎn)進(jìn)行調(diào)試,分別在攔截器,跑出異常的方法,處理異常的方法打上斷點(diǎn),測(cè)試使用response.setStatus()和response.sendError()方法來(lái)查看執(zhí)行順序,結(jié)果讓我感到驚奇:

使用response.setStatus()執(zhí)行順序:

如何在spring boot中實(shí)現(xiàn)默認(rèn)異常處理

使用response.sendError()執(zhí)行順序:

如何在spring boot中實(shí)現(xiàn)默認(rèn)異常處理

出現(xiàn)了令人驚奇兩點(diǎn):

1.setStatus()請(qǐng)求時(shí)沒(méi)有經(jīng)過(guò)攔截器
2.sendError()在異常處理完畢后經(jīng)過(guò)了一次攔截器

查看注冊(cè)攔截器配置,解決了第一個(gè)問(wèn)題的疑惑:

public void addInterceptors(InterceptorRegistry registry) {
    // 添加攔截器,去除對(duì)登錄的攔截
    registry.addInterceptor(authInterceptor)
    .excludePathPatterns("/user/login")
    .excludePathPatterns("/user/wechatLogin");
  }

這個(gè)異常是用戶登錄時(shí)拋出的,在注冊(cè)時(shí)將登錄路徑給忽略了,因?yàn)槲覀冎皇菙r截未登錄的請(qǐng)求,而請(qǐng)求登錄的請(qǐng)求不應(yīng)該攔截,這是正確的,但第二點(diǎn)卻怎么也不明白,本應(yīng)忽略攔截的請(qǐng)求,為什么換了sendError()方法后,卻在異常處理完畢后經(jīng)過(guò)了異常攔截器?

springboot的默認(rèn)異常處理

對(duì)比兩個(gè)方法的不同:setStatus()只是改了一下狀態(tài)嗎,而sendError()還有請(qǐng)求錯(cuò)誤的意味,于是猜想是不是請(qǐng)求錯(cuò)誤才會(huì)出現(xiàn)這種情況,將方法直接改為throw new RuntimeException()(沒(méi)有處理異常),發(fā)現(xiàn)攔截器攔截的請(qǐng)求的url居然是一個(gè)/error的url.

如何在spring boot中實(shí)現(xiàn)默認(rèn)異常處理

這個(gè)/error的url并未在項(xiàng)目中定義過(guò)任何的控制器中,也從未發(fā)起這樣的請(qǐng)求,上網(wǎng)一查詢,原來(lái)這是Spring Boot提供了一個(gè)默認(rèn)的映射:/error,當(dāng)處理中拋出異常之后,會(huì)轉(zhuǎn)到該請(qǐng)求中處理,并且該請(qǐng)求有一個(gè)全局的錯(cuò)誤頁(yè)面用來(lái)展示異常內(nèi)容.
但是我們的攔截器把這個(gè)請(qǐng)求攔截了(并且這個(gè)請(qǐng)求沒(méi)有攜帶正確的cookie),所以直接就返回了401錯(cuò)誤,response中也沒(méi)有我們定義的狀態(tài)碼和信息了.

json還是html

一切真相大白了,但忽然想到如果是瀏覽器發(fā)起的請(qǐng)求,服務(wù)器錯(cuò)誤后springboot默認(rèn)異常處理返回的是html,但是如果像我們前后臺(tái)分離的請(qǐng)求,返回就不應(yīng)該是html而是json的錯(cuò)誤信息了,這個(gè)要怎么區(qū)分呢?
使用google插件發(fā)送請(qǐng)求,返回的body是這樣的:

如何在spring boot中實(shí)現(xiàn)默認(rèn)異常處理

而用瀏覽器發(fā)起的請(qǐng)求返回的卻是一個(gè)html頁(yè)面:

<html>
<body>
<h2>Whitelabel Error Page</h2>
<p>This application has no explicit mapping for /error, so you are seeing this as a fallback.</p>
<div id='created'>Sat Apr 13 21:34:34 CST 2019</div>
<div>There was an unexpected error (type=Internal Server Error, status=500).</div>
<div>No message available</div>
</body>
</html>

仔細(xì)查看兩者發(fā)起的請(qǐng)求不同,在瀏覽器發(fā)起請(qǐng)求信息requestheader上發(fā)現(xiàn)了Accept字段:

Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8

關(guān)于如何在spring boot中實(shí)現(xiàn)默認(rèn)異常處理就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI