激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - spring boot 默認異常處理的實現

spring boot 默認異常處理的實現

2021-08-01 11:39鯨冬香 Java教程

這篇文章主要介紹了spring boot 默認異常處理的實現,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

本周在看陳杰寫的自定義異常的微信異常時,使用的是自定義異常狀態碼和信息,在出錯時將他拋出,并用@exceptionhandler注解定義一個全局異常處理器,根據異常的內容向前臺發送狀態碼和信息,處理異常的代碼如下圖:

?
1
2
3
4
5
6
7
//處理微信登錄的異常
  @exceptionhandler(value = wechatloginexception.class)
  public string wechatloginexceptionhandler(httpservletrequest request, httpservletresponse response, wechatloginexception e){
    logger.error("微信登錄異常:---host {} invokes url {} error: {}", request.getremotehost(), request.getrequesturl(), e.getmessage());
    response.setstatus(e.getcode());
    return e.getmessage();
  }

在這里我看的時候有點疑惑,將狀態碼寫入響應,而信息卻直接返回了,詢問陳杰,前臺果然沒有接受到e.getmessage()的信息,我上網搜索了一下,推薦他使用response.senderror(code, message)這個方法來返回異常的信息,但是這么一試之后卻遭到了奇怪的問題.

莫名的攔截器

項目配置了一個攔截器,專門用來對用戶進行驗證是否登錄的,這個是前提.在使用response.setstatus()方法時,前臺能正確的接受到傳入的狀態碼,而使用response.senderror()時,前臺卻接受到的一直是401用戶未登錄的狀態碼,打了斷點進行調試,分別在攔截器,跑出異常的方法,處理異常的方法打上斷點,測試使用response.setstatus()和response.senderror()方法來查看執行順序,結果讓我感到驚奇:

使用response.setstatus()執行順序:

spring boot 默認異常處理的實現

使用response.senderror()執行順序:

spring boot 默認異常處理的實現

出現了令人驚奇兩點:

1.setstatus()請求時沒有經過攔截器
2.senderror()在異常處理完畢后經過了一次攔截器

查看注冊攔截器配置,解決了第一個問題的疑惑:

?
1
2
3
4
5
6
public void addinterceptors(interceptorregistry registry) {
    // 添加攔截器,去除對登錄的攔截
    registry.addinterceptor(authinterceptor)
    .excludepathpatterns("/user/login")
    .excludepathpatterns("/user/wechatlogin");
  }

這個異常是用戶登錄時拋出的,在注冊時將登錄路徑給忽略了,因為我們只是攔截未登錄的請求,而請求登錄的請求不應該攔截,這是正確的,但第二點卻怎么也不明白,本應忽略攔截的請求,為什么換了senderror()方法后,卻在異常處理完畢后經過了異常攔截器?

springboot的默認異常處理

對比兩個方法的不同:setstatus()只是改了一下狀態嗎,而senderror()還有請求錯誤的意味,于是猜想是不是請求錯誤才會出現這種情況,將方法直接改為throw new runtimeexception()(沒有處理異常),發現攔截器攔截的請求的url居然是一個/error的url.

spring boot 默認異常處理的實現

這個/error的url并未在項目中定義過任何的控制器中,也從未發起這樣的請求,上網一查詢,原來這是spring boot提供了一個默認的映射:/error,當處理中拋出異常之后,會轉到該請求中處理,并且該請求有一個全局的錯誤頁面用來展示異常內容.
但是我們的攔截器把這個請求攔截了(并且這個請求沒有攜帶正確的cookie),所以直接就返回了401錯誤,response中也沒有我們定義的狀態碼和信息了.

json還是html

一切真相大白了,但忽然想到如果是瀏覽器發起的請求,服務器錯誤后springboot默認異常處理返回的是html,但是如果像我們前后臺分離的請求,返回就不應該是html而是json的錯誤信息了,這個要怎么區分呢?
使用google插件發送請求,返回的body是這樣的:

spring boot 默認異常處理的實現

而用瀏覽器發起的請求返回的卻是一個html頁面:

?
1
2
3
4
5
6
7
8
9
<html>
<body>
<h1>whitelabel error page</h1>
<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>

仔細查看兩者發起的請求不同,在瀏覽器發起請求信息requestheader上發現了accept字段:

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

頓時就明白了,在發送請求時spring-boot根據accept字段來給你返回響應的內容,例如application/json返回json,text/html返回html,真是感嘆spring-boot真是太周全了.

總結

spring-boot好心幫你默認請求異常,但是卻給你帶來了麻煩,感覺還是自己理解的不夠多,學習路還遠著呢。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:https://segmentfault.com/a/1190000018859168

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 久久艳片 | 免费a级黄色毛片 | 日韩中文一区 | av免费在线播放网址 | 黄色网址免费在线 | 久久精品成人 | 国产精品夜色视频一级区 | 噜噜噜在线 | 女人一级一级毛片 | free国产hd老熟bbw | 免看黄大片aa | 亚洲欧美在线看 | 羞羞答答xxdd在线播放 | 韩国精品视频在线观看 | 欧美成人自拍 | omofun 动漫在线观看 | 在线中文字幕网站 | 国产精品看片 | 依依成人精品视频 | 国产精品久久久久久影视 | 精品欧美一区二区精品久久 | 黄a大片| 午夜精品老牛av一区二区三区 | 国产亚洲精品影达达兔 | 欧美日韩在线看片 | 久久精品国产99久久6动漫亮点 | 国产视频在线观看一区二区三区 | 国产免费高清在线 | 久草视频国产在线 | 久久精品综合视频 | 日本一区二区不卡高清 | 日韩毛片在线看 | 日韩精品中文字幕一区 | 九九久久视频 | 日日草夜夜 | 成年免费视频黄网站在线观看 | 国产美女视频一区二区三区 | 国产91久久精品一区二区 | av最新在线观看 | 精国品产一区二区三区有限公司 | chinese hd xxxx tube|