背景
濤哥最近跟我提了個(gè)需求膀篮,要求在http請(qǐng)求的過(guò)程中直接傳輸json字符串毯焕,問(wèn)spring mvc在后端能不能直接拿到參數(shù)衍腥,一開(kāi)始筆者第一想法是在spring mvc的Controller層的方法里把@PathVariable注解去掉,就能正常讀取到傳遞過(guò)來(lái)的值纳猫,后來(lái)想想不太對(duì)婆咸,如果請(qǐng)求體里是正常的key=value形式,去掉@PathVariable是有效的芜辕,直接在body里面放個(gè)json好像不太行吧尚骄;然后濤哥把前端ajax請(qǐng)求代碼發(fā)給我看,上面指明了request的content-type是application/json侵续!
關(guān)鍵的Content-Type
平時(shí)在處理post請(qǐng)求的時(shí)候并沒(méi)有太留意倔丈,大部分post都是來(lái)自表單提交和ajax的post請(qǐng)求,默認(rèn)情況下状蜗,請(qǐng)求體里的格式都是key=value&key=value形式需五,這種編碼方式其實(shí)就是最常用的application/x-www-form-urlencoded,只是平時(shí)我們不會(huì)主動(dòng)去寫這個(gè)content-type而已轧坎;現(xiàn)在主動(dòng)指明body里面的數(shù)據(jù)以json格式傳輸宏邮,即可指明content-type為application/json。注意:jQuery的ajax方法指定的dataType=json是指明接收json格式的返回?cái)?shù)據(jù)缸血,對(duì)應(yīng)http header中accept:application/json.
$(function(){
$.ajax({
url: "http://localhost:8080/demo",
type: "POST",
dataType: "json",
contentType : "application/json",
data : JSON.stringify({username:'1', password:'2'}),
success : function(data){
console.log("success");
console.log(data);
},
error: function(e){
console.log("error");
console.log(e);
}
});
});
@RequestBody注解
一開(kāi)始并不知道Spring MVC提供了這個(gè)注解蜜氨,濤哥說(shuō)不想用這個(gè)注解,我看走眼以為是@ResponseBody捎泻,這個(gè)注解會(huì)直接把返回的對(duì)象以json形式寫在body里飒炎;而@RequestBody注解的含義就是讀取請(qǐng)求body里面的值映射成參數(shù)。之后自己動(dòng)手實(shí)驗(yàn)了一把笆豁,后端代碼如下:
@RequestMapping(value = "/demo", method = RequestMethod.POST)
@ResponseBody
public Map index(HttpServletRequest request, @RequestBody UserEntity user) {
Map<String, String> map = new HashMap<String, String>();
map.put("username", user.getUsername());
map.put("password", user.getPassword());
return map;
}
其實(shí)一開(kāi)始并不順利郎汪,總是報(bào)“415 Unsupported Media Type”這種錯(cuò)赤赊,debug輸出的log顯示不支持application/json這種media-type
[DEBUG] Resolving exception from handler [com.yuanwhy.fantasy.controller.HomeController@6ea53502]: org.springframework.web.HttpMediaTypeNotSupportedException: Content type 'application/json' not supported - org.springframework.web.servlet.handler.AbstractHandlerExceptionResolver.resolveException(AbstractHandlerExceptionResolver.java:132)
后來(lái)發(fā)現(xiàn)沒(méi)有使用<mvc:annotation-driven/>明確指明使用注解驅(qū)動(dòng),加上后就好了怒竿,所以這里<mvc:annotation-driven/>有時(shí)不加雖然也能使用@Controller這種注解砍鸠,但是最好還是一開(kāi)始就加上,否則總有一些讓人抓狂的默認(rèn)行為耕驰。
調(diào)試時(shí)控制臺(tái)打印結(jié)果顯示成功讀取到服務(wù)端返回的正確參數(shù)爷辱,其實(shí)直接從httpServletRequest里也是能讀取到body信息的,只是太麻煩而已朦肘。