@PathVariable
作用:@PathVariable
是spring3.0的一个新功能:接收请求路径中占位符的值,将URL中占位符参数{xxx}
绑定到处理器类的方法形参中@PathVariable(“xxx“)
@GetMapping("/car/{id}/owner/{username}")
public Map<String,Object> getCar(@PathVariable("id") Integer id,
@PathVariable("username") String name,
) {
Map<String,Object> map = new HashMap<>();
map.put("id",id);
map.put("name",name);
return map;
}
请求参数示例:http://localhost:8080/car/2/owner/birdy
上面的例子有多个占位符,形参列表需要定义多个参数,不是很方便,可以直接定义一个map集合 @PathVariable Map<String,String> kv
,会自动映射多个参数
@GetMapping("/car/{id}/owner/{username}")
public Map<String,Object> getCar( @PathVariable Map<String,String> kv) {
Map<String,Object> map = new HashMap<>();
String id = kv.get("id");
String name = kv.get("username");
map.put("id",id);
map.put("name",name);
return map;
}
@RequestParam
@RequestParam主要用于将请求参数区域的数据映射到控制层方法的参数上
语法:
@RequestParam(value=”参数名”,required=”true/false”,defaultValue=””)
value
:请求中传入参数的名称,即前端传过来时定义的参数名。如果不设置value值,
则前端定义的参数名必须和后端接口参数名相同required
:该参数是否为必传项。默认是true,表示请求中一定要传入对应的参数,否则会报404错误;如果设置为false,当请求中没有此参数,将会默认为null。而对于基本数据类型的变量,则必须有值,这时会抛出空指针异常。如果允许空值,则接口中变量需要使用包装类来声明。defaultValue
:参数的默认值,如果请求中没有同名的参数时,该变量默认为此值。注意默认值可以使用SpEL表达式,如"#{systemProperties['java.vm.version']}"
@GetMapping("/user")
public Map<String,Object> getUser(@RequestParam("age") Integer age,
@RequestParam("inters") List<String> inters,
) {
Map<String, Object> map = new HashMap<>();
map.put("age",age);
map.put("inters",inters);
return map;
}
请求示例:http://localhost:8080/user?age=18&inters=game&inters=music
返回结果:{"inters":["game","music"],"age":18}
@RequestParam
同样可以定义集合类型参数 @RequestParam Map<String,String> params
来映射多个参数
@RequestBody
作用:@RequestBody主要用来接收前端传递给后端的json字符串中的数据的(请求体中的数据的)
注意点:
- GET方式无请求体,所以使用@RequestBody接收数据时,前端不能使用GET方式提交数据,而是用POST方式进行提交
- 在后端的同一个接收方法里,@RequestBody与@RequestParam()可以同时使用,@RequestBody最多只能有一个,而@RequestParam()可以有多个
- 如果参数时放在请求体application/json传入后台的话,那么后台要用@RequestBody才能接收到
- 如果不是放在请求体中的话,那么后台接收前台传过来的参数时,要用@RequestParam来接收,或者形参前 什么也不写也能接收
在实际的工作需求中,写接口的人会经常遇到对于Bean各种各样的操作,比如我对数据库要写个Bean,返回前台还得有个Vo bean,在另一个需求又用到和这个类似的Bean了,但是就有些字段不一样啊,字段的有些类型不一样啊,接收的参数名不一样啊,反正就是一个小小的需求变动,可能就需要写好几个不同的Bean,之前需求没那么多的时候写两个就写两个吧,后来写的多了就烦了,所以认真研究下怎么转换就迫在眉睫了。
@JsonAlias
作用是在反序列化的时候可以让Bean的属性接收多个json字段的名称。可以加在字段上或者getter和setter方法上。
@Data
public class User implements Serializable{
@JsonAlias({"name","user"})
private String username;
private String password;
private Integer age;
}
实体类需要有get和set方法
@PostMapping("/user")
public User getUserIfo(@RequestBody User user){
return user;
}
路径
:http://localhost:8080/user
//入参
{
“name” : “小明”,
“password” : “123”,
"age" : 15
}
可以从下面看到json字段是name也成功对应到了Bean的username属性 ,但是字段名没变
//postman返回结果
{
"username": "birdy",
"password": "123",
"age": 18
}
@JsonProperty
向流中写入对象的操作 称为 序列化
从流中读取对象的操作 成为 反序列化
这个注解是更改Bean字段的属性名用的。
- access:是更改逻辑属性序列化和反序列化的可见性,
Access.WRITE_ONLY:只在序列化时使用修改后的字段
Access.READ_ONLY:只在反序列化时使用,类似于@JsonAlias注解
Access.READ_WRITE:在序列化和反序列化都使用修改后字段
Access.AUTO:自动确定,一般是和第三个一样,啥情况不一样我也不清楚,如果不写access,默认就是这个。
value是逻辑属性的名称,如果只有value则省略
- value::更改后的属性名
示例
@Data
public class User implements Serializable {
//只在序列化写入操作时使用该属性名
@JsonProperty(value = "name", access = JsonProperty.Access.WRITE_ONLY)
private String username;
private String password;
private Integer age;
}
@PostMapping("/user")
public User getUserIfo(@RequestBody User user){
System.out.println(user.toString());
return user;
}
入参:
{
"name" : "birdy",
"password" : "123",
"age" : 18
}
结果:
//console
User(username=birdy, password=123, age=18)
//postman没有获取到name属性
{
"password": "123",
"age": 18
}
查看源码
Access setting that means that the property may only be written (set) as part of deserialization (using "setter" method, or assigning to Field
表示只有在写入时调用属性的set方法才会用该属性名
猜猜若改成@JsonProperty(value = "name", access = JsonProperty.Access.READ_ONLY)
会怎样?
//控制台
User(username=null, password=123, age=18)
//postman
{
"password": "123",
"age": 18,
"name": null
}
由于写入时无法调用set方法赋值,所以控制台打印该属性时没有值;但是返回结果已经可以调用修改后的属性get方法,所以username属性已经被修改为name,只是没有值
若是access = JsonProperty.Access.AUTO
//控制台
User(username=birdy, password=123, age=18)
//postman
{
"password": "123",
"age": 18,
"name": birdy
}
可以看到控制台打印的属性名称都没有变,access只影响属性在序列化和反序列化时的名称