如何應對spring框架的HTTP ERROR 400 Bad Request錯誤返回問題
HTTP ERROR 400 Bad Request 產(chǎn)生的流程
在使用springmvc相關(guān)的框架的時候,調(diào)用接口經(jīng)常產(chǎn)生以下錯誤

這種錯誤往往在info的log級別下看不到日志,難以排查,我們先來模擬錯誤產(chǎn)生的其中一個原因
從spring框架的入口FrameworkServlet開始找
1.假設(shè)我們的接口使用get請求,則從doGet里面的processRequest方法開始調(diào)式,并且假設(shè)模擬一個請求類型不匹配的參數(shù),比如需要一個int類型的參數(shù)但是傳一個字符串。


2.下一步進入doDispatch方法

3.進入接口對應方法的處理,使用handler處理

4.進入到實際調(diào)用的方法的代碼,invokeHandleMethod,在調(diào)用此方法的時候,會先做一些參數(shù)的校驗,如是否空值(MissingServletRequestParameterException)、是否類型匹配(TypeMismatchException)、媒體類型是否支持(HttpMediaTypeNotSupportedException)等

5.進入invokeForRequest方法

6.進入getMethodArgumentValues方法,此方法用來獲取處理接口實際方法的參數(shù)和值的對應關(guān)系。
可以看到在這個地方產(chǎn)生了錯誤,并且向上一級拋出了異常且異常被doDispatch捕獲,然后進入processDispatchResult方法處理返回的結(jié)果


7.調(diào)用HandlerExecutionChain的異常處理方法去處理異常

8.調(diào)用resolveException處理,處理的時候是循環(huán)使用List<HandlerExceptionResolver> resolvers中的所有的handlerExceptionResolver依次處理,如果處理成功則返回。
可以看到一共有5個handlerExceptionResolver,本次接口調(diào)用被第二個handlerExceptionResolver處理成功并返回


9.異常被成功處理

10.返回錯誤碼和空ModelAndView

11.在request的Attribute屬性中設(shè)置異常信息

12.進行handler的后置處理,比如執(zhí)行HandlerInterceptor的afterCompletion方法

13.執(zhí)行processRequest的finally中的剩余代碼邏輯

HTTP ERROR 400 Bad Request 產(chǎn)生的原因
由于在以上代碼中調(diào)用了
response.sendError(HttpServletResponse.SC_BAD_REQUEST); return new ModelAndView();
導致產(chǎn)生了400頁面
400錯誤為什么不能被自定義的異常處理器捕獲并且處理
由于handlerExceptionResolvers要按順序依次執(zhí)行并且執(zhí)行成功后就立即返回后續(xù)的將不再執(zhí)行,我們自定義的handlerExceptionResolvers排在List的最后(由上圖可以很清晰的知道,我們自定義的WebExceptionHandler排在最后),所以,輪不到它處理,spring就內(nèi)部就將異常消化處理好了。
為什么不能自己定制輸出內(nèi)容
response.sendError(HttpServletResponse.SC_BAD_REQUEST)
上面的代碼已經(jīng)使用了response將結(jié)果寫入到輸出流,所以后續(xù)將無法使用response或者類似于
PrintWriter out = null;
try {
out = response.getWriter();
Map<String,Object> result = new HashMap<String, Object>();
result.put("code",be.getErrorCode());
result.put("message",be.getMessage());
out.write(JSONObject.toJSONString(result));
out.flush();
} catch (IOException e) {
} finally {
if (out != null) {
out.close();
out = null;
}
}
的方式自定義輸出
目前可以做的
研究了一番,發(fā)現(xiàn)不能自定義輸出(或者有其他的好方法沒發(fā)現(xiàn)),所以只能記錄下錯誤日志并且打印出來,這樣就不會稀里糊涂了
未加日志之前

加日志之后


如何加日志
通過以上的代碼分析可知,spring會將異常寫入到request的Attribute屬性中,所以只需要
Object attribute = request.getAttribute(DispatcherServlet.class.getName() + ".EXCEPTION");
就可以獲得異常信息了,并且在自己的Filter里面打印,這樣就不會在產(chǎn)生400之類類似的錯誤的時候一頭霧水了
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
解決mybatis中resultType取出數(shù)據(jù)順序不一致的問題
這篇文章主要介紹了解決mybatis中resultType取出數(shù)據(jù)順序不一致的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02
Java中tomcat memecached session 共享同步問題的解決辦法
這篇文章主要介紹了Java中tomcat memecached session 共享同步問題的解決辦法的相關(guān)資料,需要的朋友可以參考下2015-10-10
java優(yōu)先隊列PriorityQueue中Comparator的用法詳解
這篇文章主要介紹了java優(yōu)先隊列PriorityQueue中Comparator的用法詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-02-02

