@RequestBody 与 @RequestParam
最近赶项目中,时不时在Ajax传数据的时候卡一会,这次下定决心总结一下。
先介绍一下基础知识,还是以遇到的实例进行说明,希望以后能举一反三。
介绍
相信 @RequestParam 已经用的很多了,就不多说了,主要说一下 @RequestBody
@RequestBody主要用来接收前端传递给后端的json字符串中的数据(请求体中的数据的);GET方式无请求体,所以使用@RequestBody接收数据时,前端不能使用GET方式提交数据,而是用POST方式进行提交。在后端的同一个接收方法里,@RequestBody()与@RequestParam()可以同时使用,@RequestBody最多只能有一个,而@RequestParam()可以有多个。
一个请求,只有一个RequestBody;一个请求,可以有多个RequestParam。
当同时使用 @RequestParam 和 @RequestBody 时,@RequestParam 指定的参数可以是普通元素、数组、集合、对象等等(即,当 @RequestBody 与 @RequestParam 可以同时使用时,原 SpringMVC 接收参数的机制不变,只不过 RequestBody 接收的是请求体里面的数据;而 RequestParam 接收的是 key-value 里面的参数,所以它会被切面进行处理从而可以用普通元素、数组、集合、对象等接收)。
如果参数前不写 @RequestParam(xx) 的话,那么就前端可以有可以没有对应的xx名字才行,如果有xx名的话,那么就会自动匹配;没有的话,请求也能正确发送。
如果后端参数是一个对象,且该参数前是以@RequestBody修饰的,那么前端传递json参数时,必须满足以下要求:
- 后台 @RequestBody 注解对应的类在将 HTTP 的输入流(含请求体)装配到目标类(即:@RequestBody 后面的类)时,会根据 json 字符串中的 key 来匹配对应实体类的属性,如果匹配一致且 json 中的该 key 对应的值符合(或可转换为),这里会在下面详细分析,实体类的对应属性的类型要求时,会调用实体类的setter方法将值赋给该属性。
- json 字符串中,如果value为 "" 的话,后端对应属性如果是 String 类型的,那么接受到的就是 "",如果是后端属性的类型是 Integer、Double 等类型,那么接收到的就是 null。
- json 字符串中,如果 value 为 null 的话,后端对应收到的就是 null。
- 如果某个参数没有 value 的话,在传 json 字符串给后端时,要么干脆就不把该字段写到 json 字符串中;要么写 value 时,必须有值,null 或 "" 都行。
实例说明
User实体类:
public class User {
/** 姓名 */
private String name;
/** 年龄 */
private String age;
/** 性别 */
private String gender;
/** 备注 */
private String remark;
//getter,setter等省略...
}
Team实体类:
public class Team {
/** id */
private Integer team;
/** 名字 */
private String teamName;
/** 性别 */
private List<String> honors;
/** 备注 */
private List<User> teamMembers;
//getter,setter等省略...
}
@RequestBody 直接以 String 接收前端传过来的 json 数据:
后端对应的 Controller:
/**
* 直接用String接收json数据
* @param jsonString
* @return
*/
@RequestMapping("mytest0")
public String test(@RequestBody String jsonString){
System.out.println(jsonString);
return jsonString;
}
@RequestBody以简单对象接收前端传过来的json数据:
后端对应的 Controller:
/**
* 用简单User对象接收json数据
* @param user
* @return user.toString()
*/
@RequestMapping("mytest1")
public String test(@RequestBody User user){
System.out.println(user.toString());
return user.toString();
}
使用PostMan测试:
@RequestBody 以复杂对象接收前端传过来的 json 数据:
后端对应的 Controller:
/**
* 用复杂Team对象接收json数据
* @param tean
* @return
*/
@RequestMapping("mytest2")
public String test(@RequestBody Team team){
System.out.println(team.toString());
return team.toString();
}
使用PostMan测试:
@RequestBody 与简单的 @RequestParam 同时使用:
后端对应的 Controller:
/**
* @RequestBody() 与简单的 @RequestParam() 同时使用,这里的对象用数组也ok
* @param tean
* @return
*/
@RequestMapping("/register/addStudent")
@ResponseBody
public ResponseData addP(@RequestBody RegisterStudent [] data,
@RequestParam("registerName") String registerName)
看下前端的Ajax:
var data = [];//新建数组
$("form").each(function () {
data.push($(this).serializeObject());
});
$.ajax({
type:"POST",
url: "/register/addStudent?registerName="+name,
dataType: "json",
contentType: "application/json", //这里很重要
data: JSON.stringify(data), //只有这一个参数,json格式,后台解析为实体,后台可以直接用
success: function(data){
if (data.code === 200) {
$.toast("提交成功");
} else {
$.toast("内部错误",data.msg);
}
}
});
对象数组+额外参数,这应该算是最复杂的吧。。。
@RequestBody接收请求体中的json数据;不加注解接收URL中的数据并组装为对象:
后端对应的Controller:
/**
* @RequestBody()装配请求
* 第二个参数不加注解,装配 url 中的参数信息
* @param user
* @return
*/
@RequestMapping("mytest4")
public String test(@RequestBody User user1, User user2){
System.out.println(user1.toString());
System.out.println(user2.toString());
return user1.toString() + "\n" + user2.toString();
}
使用PostMan测试:
目前就这样,收工。