关于前后端数据交互与HTTP请求的关系
写本篇文章的目的是想让更多前后端联调的工作变得更清晰明了
HTTP请求的简单认识
资源地址
协议://域名或者ip:端口/路径(protocol://host:port/address)
如 ""
https默认的端口为443,http默认的端口为80
请求方式
方式:get/post/put/patch/delete/head/options/connect/trace,
根据restful风格,我们一般最多用到get/post/put/patch/delete
- get代表从服务器获取资源
- post代表为服务器添加资源
- put/patch代表修改服务器资源,put更新资源的全部,patch根据资源的局部
- delete代表删除服务器资源
请求参数
请求参数我们一般成为parameter(简称param),它是url问号后面'?key1=value1&key2=value2'的部分,
一般代表请求资源的条件
请求头
请求头我们称为header,一般是一个map结构,记录请求的一些信息,如Cookie和User-Agent
请求体
请求体我们称为body,一般存放的是传递给后台的数据,
请求头中有个content-type字段去标识body中数据的编码方式
- application/x-www-form-urlencoded是表单key-value编码方式
- application/json是json字符串编码方式
- ...
请求方式与param和body之间的关系
本质上来说,请求方式与以何种方式发送数据是没有必然关系的
GET请求既可以发送param数据也可以发送body数据,同理POST及其他请求方式也都可以
但是,一般来说,GET请求方式,参数是规定要放在param中的。
GET请求在某些HTTP请求客户端是无法携带body的,比如java原生java.net.URLConnection,
OkHttpClient,Apache HttpClient和前端jquery ajax都不支持。
但是Android端AsyncHttpClient包和PostMan工具是可以支持GET请求携带body的
怎样进行前后端数据交互
后端以 SpringMVC/SpringBoot 举例
//数据模型Person类
@Data//lombok
pulic class Person {
private String id;
private String name;
private String address;
private Integer age;
}
//后端接口,为了直观,省略状态,直接返回bean
@RestController
@RequestMapping("/person")
public class PersonController {
//Person的Service
@Autowired
private PersonService personService;
//分页查询
@GetMapping("/queryByParam")
public List<Person> getStudent(Person person,
@RequestParam(defaultValue = "1") int pageNum,
@RequestParam(defaultValue = "10") int pageSize){
//根据person中不为null的属性查询到的Person列表
// pageNum和pageSize为分页条件,分别代表第几页和一页条数
List<Person> personList = personService.page(person,pageNum,pageSize);
return personList;
}
//新增数据
@PostMapping("/save")
public Person getStudent(@RequestBody Person person){
//往数据库中插入一条记录
personService.insert(person);
return person;
}
}
前端以 JQuery AJAX 举例
//GET分页查询
let pageNum = 1,pageSize = 10;//也可以作为data对象的属性,这里分开
let data = {name:'张三',address:'上海'};
$.ajax({
type : "GET",
url : `http://host:port/person/queryByBody?pageNum=${pageNum}&pageSize=${pageSize}`,
data : data,
dataType:"json",//返回数据类型
success : function(result) {//成功回调
//成功处理
},
error : function(err){//请求失败
//失败处理
}
});
//POST新增数据
let person = {
id:"1001",
name:"李四",
address:"北京",
age:18
};
$.ajax({
type : "POST",
url : "http://host:port/person/save",
data : JSON.stringify(person),
dataType:"json",//返回数据类型
contentType: "application/json",//请求的媒体类型
success : function(result) {//成功回调
//成功处理
},
error : function(err){//请求失败
//失败处理
}
});
后端Controller层各注解使用(遵循restful风格)
Controller类注解
- Controller类上使用 @RestController 注解,等价与 @Controller + @ResponseBody
- Controller类上最好加上一个 @RequestMapping("/xxx") 代表父级路径
Controller方法注解
映射地址注解
- @GetMapping 等价于 @RequestMapping(method=RequestMethod.GET) ,代表GET请求数据
- @PostMapping 等价于 @RequestMapping(method=RequestMethod.POST) ,代表POST新增数据
- @PutMapping 等价于 @RequestMapping(method=RequestMethod.PUT) ,代表PUT修改数据
- @DeleteMapping 等价于 @RequestMapping(method=RequestMethod.DELETE) ,代表DELETE删除数据
参数注解
- @RequestParam
该注解是用于获取在GET方式中QueryString/param的值,和Content-Type为application/x-www-form-urlencoded的post方式中body data的值。
等同于 javax.servlet.httpHttpServletRequest的getParameter(String name)方法。其中注解参数name/value代表getParameter中的name,
默认值为参数名;required代表参数是否为必须,默认值为true;defaultValue代表参数的默认值
//使用场景一:url中param的key与方法参数名相同且name不是必须时,可省略
@GetMapping("/get")
public Response get(String name){}
//使用场景二:url中param的key与方法参数名相同,但name是必须时,不可省略
@GetMapping("/get")
public Response get(@RequestParam String name){}
//使用场景三:url中param的key与方法参数名不同且不是必须,不可省略
@GetMapping("/get")
public Response get(@RequestParam(name="username",required=false) String name){}
- @PathVariable
该注解代表参数为路径上的变量
// url为 '/get/1' 时,id="1"
@GetMapping("/get/{id}")
public Response get(@PathVariable String id){}
@GetMapping("/get/{uid}")
public Response get(@PathVariable("uid") String id){}
- @RequestBody
前端发送body数据且contentType为application/json时使用,一般以一个java bean类型去接收,若以String类型接收,则接收到的是json字符串
// jquery ajax 需要使用 JSON.stringify 序列化数据
@PostMapping("/post")
public Response post(@RequestBody Person person){}
不积跬步无以至千里