如何應(yīng)對spring框架的HTTP ERROR 400 Bad Request錯(cuò)誤返回問題
HTTP ERROR 400 Bad Request 產(chǎn)生的流程
在使用springmvc相關(guān)的框架的時(shí)候,調(diào)用接口經(jīng)常產(chǎn)生以下錯(cuò)誤
這種錯(cuò)誤往往在info的log級別下看不到日志,難以排查,我們先來模擬錯(cuò)誤產(chǎn)生的其中一個(gè)原因
從spring框架的入口FrameworkServlet開始找
1.假設(shè)我們的接口使用get請求,則從doGet里面的processRequest方法開始調(diào)式,并且假設(shè)模擬一個(gè)請求類型不匹配的參數(shù),比如需要一個(gè)int類型的參數(shù)但是傳一個(gè)字符串。
2.下一步進(jìn)入doDispatch方法
3.進(jìn)入接口對應(yīng)方法的處理,使用handler處理
4.進(jìn)入到實(shí)際調(diào)用的方法的代碼,invokeHandleMethod,在調(diào)用此方法的時(shí)候,會(huì)先做一些參數(shù)的校驗(yàn),如是否空值(MissingServletRequestParameterException)、是否類型匹配(TypeMismatchException)、媒體類型是否支持(HttpMediaTypeNotSupportedException)等
5.進(jìn)入invokeForRequest方法
6.進(jìn)入getMethodArgumentValues方法,此方法用來獲取處理接口實(shí)際方法的參數(shù)和值的對應(yīng)關(guān)系。
可以看到在這個(gè)地方產(chǎn)生了錯(cuò)誤,并且向上一級拋出了異常且異常被doDispatch捕獲,然后進(jìn)入processDispatchResult方法處理返回的結(jié)果
7.調(diào)用HandlerExecutionChain的異常處理方法去處理異常
8.調(diào)用resolveException處理,處理的時(shí)候是循環(huán)使用List<HandlerExceptionResolver> resolvers
中的所有的handlerExceptionResolver依次處理,如果處理成功則返回。
可以看到一共有5個(gè)handlerExceptionResolver,本次接口調(diào)用被第二個(gè)handlerExceptionResolver處理成功并返回
9.異常被成功處理
10.返回錯(cuò)誤碼和空ModelAndView
11.在request的Attribute屬性中設(shè)置異常信息
12.進(jìn)行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();
導(dǎo)致產(chǎn)生了400頁面
400錯(cuò)誤為什么不能被自定義的異常處理器捕獲并且處理
由于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)),所以只能記錄下錯(cuò)誤日志并且打印出來,這樣就不會(huì)稀里糊涂了
未加日志之前
加日志之后
如何加日志
通過以上的代碼分析可知,spring會(huì)將異常寫入到request的Attribute屬性中,所以只需要
Object attribute = request.getAttribute(DispatcherServlet.class.getName() + ".EXCEPTION");
就可以獲得異常信息了,并且在自己的Filter里面打印,這樣就不會(huì)在產(chǎn)生400之類類似的錯(cuò)誤的時(shí)候一頭霧水了
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
解決mybatis中resultType取出數(shù)據(jù)順序不一致的問題
這篇文章主要介紹了解決mybatis中resultType取出數(shù)據(jù)順序不一致的問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02Java中tomcat memecached session 共享同步問題的解決辦法
這篇文章主要介紹了Java中tomcat memecached session 共享同步問題的解決辦法的相關(guān)資料,需要的朋友可以參考下2015-10-10java優(yōu)先隊(duì)列PriorityQueue中Comparator的用法詳解
這篇文章主要介紹了java優(yōu)先隊(duì)列PriorityQueue中Comparator的用法詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02