SpringMVC @RequestBody出現(xiàn)400 Bad Request的解決
@RequestBody出現(xiàn)400 Bad Request的問題
今天與同事調(diào)試一個接口,發(fā)現(xiàn)后臺使用@RequestBody老是獲取不到數(shù)據(jù)。查了網(wǎng)上很多資料,要使用@RequestBody來轉(zhuǎn)換JSON字符串為對象
大概是以下幾個點
1. 請求的Content-Type要是application/json
2. 請求的類型要是POST
3. 前臺json傳遞的key在后臺的實體對象中存在,也就是JSON要與實體對象要對應(yīng),并且名稱要一致(如果不一致可以使用@JsonProperty來做映射)
以上這些我基本都檢查過了,都沒有問題,但是程序根本就不會進入接口方法,詳細檢查了前端傳送到后臺確實 application/json類型的數(shù)據(jù)。
如下:
POST /quickstart/api/v1/dataSync/syncUser HTTP/1.1 Content-Length: 67 Host: 192.168.1.231:9090 Content-Type: application/json {"username": "1111","password": "e10adc3949ba59abbe56e057f20f883e"} HTTP/1.1 400 Bad Request Content-Length: 0 Server: Jetty(7.6.15.v20140411)
SpringMVC也沒有拋出任何錯誤,于是在接口Controller加入以下異常處理代碼來確定異常信息:
@ResponseBody ?? ?@ResponseStatus(HttpStatus.BAD_REQUEST) ?? ?@ExceptionHandler(HttpMessageNotReadableException.class) ?? ?public void messageNotReadable(HttpMessageNotReadableException exception, HttpServletResponse response){ ?? ??? ?//調(diào)試作用,用來調(diào)試前臺傳遞json后臺無法正確映射問題 ?? ??? ? ?? ??? ?logger.error("請求參數(shù)不匹配。", exception); ?? ? ? ? ?? ?}
終于后臺打印出錯誤信息如下:
Caused by: com.fasterxml.jackson.databind.JsonMappingException: No content to map due to end-of-input
at [Source: org.eclipse.jetty.server.HttpInput@42f0d7f3; line: 1, column: 1]
at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:164) ~[jackson-databind-2.4.0.jar:2.4.0]
at com.fasterxml.jackson.databind.ObjectMapper._initForReading(ObjectMapper.java:3095) ~[jackson-databind-2.4.0.jar:2.4.0]
at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:3036) ~[jackson-databind-2.4.0.jar:2.4.0]
但是還是找不到問題所在,最后無意發(fā)現(xiàn)前面有一段測試代碼(因為程序一直無法進入Controller的接口方法,所以我在Spring攔截器中加了一段代碼),測試前端傳過來的JSON數(shù)據(jù)的內(nèi)容。也就是這段代碼導(dǎo)致了這一切。
@Override ?? ?public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { ?? ??? ?BufferedReader reader = new BufferedReader(new InputStreamReader(request.getInputStream())); ?? ??? ?StringBuffer buffer = new StringBuffer(); ? ?? ??? ?String line = " "; ? ?? ??? ?while ((line = reader.readLine()) != null){ ? ?? ??? ? ? ? buffer.append(line); ? ?? ??? ?} ? ?? ??? ?System.out.println(buffer.toString());
查看文檔,發(fā)現(xiàn)HttpServletRequest.getInputStream(),一般只能被調(diào)用一次,在request.getinputstream讀取一次后position到了文件末尾,第二次就讀取不到數(shù)據(jù),由于無法reset(),所以,request.getinputstream只能讀取一次。
做了幾年程序第一次知道HttpServletRequest.getInputStream()只能被讀取一次,真是學(xué)藝不精啊。
post 400 (Bad Request)異常怎么排查參數(shù)
問題描述
用ajax請求時報post 400 (Bad Request)的異常,前臺js參數(shù)JSON.stringify(data),后臺controller 中@RequestBody XX xx(javabean)接收參數(shù)。
通常發(fā)生400時,即使在后臺方法上設(shè)置斷點,但因400是參數(shù)由json轉(zhuǎn)換成Javabean時發(fā)生異常,所以不會進入斷點。
此時如果單靠眼力一一排查參數(shù)中的每個值,簡直能把眼瞅瞎……還未必能找到……
解決辦法
辦法一:
從大牛那取經(jīng),可以org.springframework.web.servlet.DispatcherServlet中設(shè)置斷點,好像是doService方法,下次再遇到自己打算試試。
辦法二:
自己找到了一個low但比較簡單直觀的辦法,哈哈
1.瀏覽器F12,調(diào)試狀態(tài)下在network/網(wǎng)絡(luò)找到400(Bad Request)的請求,復(fù)制出該post請求的json格式的請求參數(shù)。
以chrome為例,點擊“view source”會顯示json字符串格式的參數(shù)。
2.在后臺方法中將該json字符串轉(zhuǎn)換成JSONObject,再將JSONObject轉(zhuǎn)換成實體XX。
代碼:
com.alibaba.fastjson.JSONObject.toJavaObject( (com.alibaba.fastjson.JSONObject)(com.alibaba.fastjson.JSONObject.parse("{'ts':1493184921039,'pk':nulll}")),XX.class)
其中的{'ts':1493184921039,'pk':nulll}是請求參數(shù)字符串(將雙引號改成單引號),JSONObject.parse()方法將其轉(zhuǎn)換成JSONObject格式;JSONObject.toJavaObject()是轉(zhuǎn)換為實體XX的方法。
3.運行代碼,就會報錯,錯誤中會提示那些參數(shù)有問題,進而修改就可以啦。
到目前為止遇到過兩次參數(shù)的問題,一次是時間戳ts的格式導(dǎo)致400的,當(dāng)時ts是是由net.sf.json.JSONObject處理過返回給前臺頁面的,等到再將包含該ts的實體json串傳給后臺時就因無法轉(zhuǎn)換為實體報錯了。
第二次就是現(xiàn)在因為實體中的一個屬性是空值,而實體類中該屬性的set方法對該屬性值進行了特殊處理,卻又未判空導(dǎo)致了空指針異常。
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
使用JMX監(jiān)控Zookeeper狀態(tài)Java API
今天小編就為大家分享一篇關(guān)于使用JMX監(jiān)控Zookeeper狀態(tài)Java API,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-03-03深入學(xué)習(xí)java8?中的CompletableFuture
本文主要介紹了java8中的CompletableFuture,CompletableFuture實現(xiàn)了CompletionStage接口和Future接口,前者是對后者的一個擴展,增加了異步回調(diào)、流式處理、多個Future組合處理的能力,使Java在處理多任務(wù)的協(xié)同工作時更加順暢便利,下文需要的朋友可以參考一下2022-05-05Maven腳手架如何基于jeecg實現(xiàn)快速開發(fā)
這篇文章主要介紹了Maven腳手架如何基于jeecg實現(xiàn)快速開發(fā),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-10-10詳解log4j-over-slf4j與slf4j-log4j12共存stack overflow異常分析
這篇文章主要介紹了詳解log4j-over-slf4j與slf4j-log4j12共存stack overflow異常分析,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-07-07基于RecyclerChart的KLine繪制Volume實現(xiàn)詳解
這篇文章主要為大家介紹了基于RecyclerChart的KLine繪制Volume實現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-03-03基于Java解決華為機試實現(xiàn)整數(shù)與IP地址間的轉(zhuǎn)換?
這篇文章主要介紹了基于Java解決華為機試實現(xiàn)整數(shù)與IP地址間的轉(zhuǎn)換,文章舉例說明圍繞文章主題展開相關(guān)內(nèi)容,具有一定的參考價值,需要的小伙伴可以參考一下2022-02-02