什么是 RESTful API
官网地址:https://restfulapi.cn/
- REST 是 Representational State Transfer 的缩写,如果一个架构符合 REST 原则,就称它为 RESTful 架构;
- RESTful 架构可以充分的利用 HTTP 协议的各种功能,是 HTTP 协议的最佳实践;
- RESTful API 是一种软件架构风格、设计风格,可以让软件更加清晰,更简洁,更有层次,可维护性更好;
API 请求
HTTP 动词
GET: 读取(Read)
POST: 新建(Create)
PUT: 更新(Update)
PATCH: 更新(Update),通常是部分更新
DELETE:删除(Delete)
请求设计
请求 = 动词 + 宾语
- 动词 使用五种 HTTP 方法,对应 CRUD 操作;
- 宾语 URL 应该全部使用名词复数,可以有例外,比如搜索可以使用更加直观的 search;
- 过滤信息(Filtering) 如果记录数量很多,API 应该提供参数,过滤返回结果。 ?limit=10 指定返回记录的数量 ?offset=10 指定返回记录的开始位置;
API 响应
状态码
五大类状态码,总共100多种,覆盖了绝大部分可能遇到的情况。每一种状态码都有约定的解释,客户端只需查看状态码,就可以判断出发生了什么情况。API 不需要1xx状态码;
1xx:相关信息
2xx:操作成功
3xx:重定向
4xx:客户端错误
5xx:服务器错误
响应设计
服务器回应数据
- 客户端的每一次请求,服务器都必须给出回应。回应包括 HTTP 状态码和数据两部分;
- 客户端请求时,要明确告诉服务器,接受 JSON 格式,请求的 HTTP 头的 ACCEPT 属性要设成 application/json
- 服务端返回的数据,不应该是纯文本,而应该是一个 JSON 对象。服务器回应的 HTTP 头的 Content-Type 属性要设为 application/json;
SpringBoot 快速实践 RESTful
注解实现
Srping Boot 提供了与 Rest 操作方式(GET、POST、PUT、DELETE)对应的注解:
HTTP 动词 | Srping Boot 注解 | 说明 |
GET | @GetMapping | 读取 |
POST | @PostMapping | 新建 |
PUT | @PutMapping | 更新 |
PATCH | @PatchMapping | 更新,通常是部分更新 |
DELETE | @DeleteMapping | 删除 |
实践 RESTful
根据 RESTful 风格设计编写用户操作的 RESTful API。
1.定义用户类
package com.csp.mingyue.api.model;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.ToString;
/** @author Strive */
@Data
@ToString
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class MingYueUser {
private Long userId;
private String username;
}
2.编写 Service
package com.csp.mingyue.api.service;
import cn.hutool.core.map.MapUtil;
import com.csp.mingyue.api.model.MingYueUser;
import java.util.Map;
import org.springframework.stereotype.Service;
/** @author Strive */
@Service
public class MingYueUserService {
/** 模拟用户存储 */
private static final Map<Long, MingYueUser> USER_MAP = MapUtil.newHashMap();
static {
USER_MAP.put(1L, MingYueUser.builder().userId(1L).username("mingyue").build());
}
/**
* 根据用户ID查询用户信息
*
* @param userId 用户ID
* @return 用户信息
*/
public MingYueUser queryUserById(Long userId) {
return USER_MAP.get(userId);
}
/**
* 添加用户
*
* @param user 用户信息
* @return 新用户ID
*/
public Long addUser(MingYueUser user) {
USER_MAP.put(2L, user);
return user.getUserId();
}
/**
* 更新用户
*
* @param user 用户信息
*/
public String updateUser(MingYueUser user) {
MingYueUser mingYueUser = USER_MAP.get(user.getUserId());
USER_MAP.put(user.getUserId(), user);
return mingYueUser.getUsername() + " 更新为:" + user.getUsername();
}
/**
* 根据用户ID删除用户
*
* @param userId 用户ID
*/
public String deleteUser(Long userId) {
int size = USER_MAP.size();
USER_MAP.remove(userId);
return "原" + size + "用户,删除后还有" + USER_MAP.size();
}
}
3.编写 Controller
package com.csp.mingyue.api.controller;
import com.csp.mingyue.api.model.MingYueUser;
import com.csp.mingyue.api.service.MingYueUserService;
import lombok.RequiredArgsConstructor;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.PutMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/** @author Strive */
@RestController
@RequiredArgsConstructor
@RequestMapping("/user")
public class MingYueUserController {
private final MingYueUserService mingYueUserService;
@GetMapping("/{userId}")
public ResponseEntity<MingYueUser> queryUserById(@PathVariable Long userId) {
return ResponseEntity.ok(mingYueUserService.queryUserById(userId));
}
@PostMapping
public ResponseEntity<Long> addUser(@RequestBody MingYueUser user) {
return ResponseEntity.ok(mingYueUserService.addUser(user));
}
@PutMapping
public ResponseEntity<String> updateUser(@RequestBody MingYueUser user) {
return ResponseEntity.ok(mingYueUserService.updateUser(user));
}
@DeleteMapping("/{userId}")
public ResponseEntity<String> deleteUser(@PathVariable Long userId) {
return ResponseEntity.ok(mingYueUserService.deleteUser(userId));
}
}
4.测试接口
下面的 Json 数据复制进入 Json 文件,
Apifox
导入类型选择Openapi/Swagger
即可测试接口,或者直接码云下载 mingyue-springboot-api Demo
{
"openapi": "3.0.1",
"info": {
"title": "mingyue-springboot-learning",
"description": "",
"version": "1.0.0"
},
"tags": [
{
"name": "mingyue-springboot-api"
}
],
"paths": {
"/user/{userId}": {
"get": {
"summary": "根据用户ID获取用户信息",
"x-apifox-folder": "mingyue-springboot-api",
"x-apifox-status": "developing",
"deprecated": false,
"description": "",
"tags": [
"mingyue-springboot-api"
],
"parameters": [
{
"name": "userId",
"in": "path",
"description": "",
"required": true,
"example": "1",
"schema": {
"type": "string"
}
}
],
"responses": {
"200": {
"description": "成功",
"content": {
"application/json": {
"schema": {
"type": "object",
"properties": {}
},
"examples": {}
}
}
}
}
},
"delete": {
"summary": "根据用户ID删除用户",
"x-apifox-folder": "mingyue-springboot-api",
"x-apifox-status": "developing",
"deprecated": false,
"description": "",
"tags": [
"mingyue-springboot-api"
],
"parameters": [
{
"name": "userId",
"in": "path",
"description": "",
"required": true,
"example": "1",
"schema": {
"type": "string"
}
}
],
"responses": {
"200": {
"description": "成功",
"content": {
"application/json": {
"schema": {
"type": "object",
"properties": {}
},
"examples": {}
}
}
}
}
}
},
"/user": {
"put": {
"summary": "更新用户",
"x-apifox-folder": "mingyue-springboot-api",
"x-apifox-status": "developing",
"deprecated": false,
"description": "",
"tags": [
"mingyue-springboot-api"
],
"parameters": [
{
"name": "Content-Type",
"in": "header",
"description": "",
"required": false,
"example": "{{'application/json'}}",
"schema": {
"type": "string"
}
}
],
"requestBody": {
"content": {
"text/plain": {
"schema": {
"type": "string"
},
"example": "{\"userId\":2,\"username\":\"Strive Update\"}"
}
}
},
"responses": {
"200": {
"description": "成功",
"content": {
"application/json": {
"schema": {
"type": "object",
"properties": {}
},
"examples": {}
}
}
}
}
},
"post": {
"summary": "添加用户",
"x-apifox-folder": "mingyue-springboot-api",
"x-apifox-status": "developing",
"deprecated": false,
"description": "",
"tags": [
"mingyue-springboot-api"
],
"parameters": [
{
"name": "Content-Type",
"in": "header",
"description": "",
"required": false,
"example": "{{'application/json'}}",
"schema": {
"type": "string"
}
}
],
"requestBody": {
"content": {
"text/plain": {
"schema": {
"type": "string"
},
"example": "{\"userId\":2,\"username\":\"Strive\"}"
}
}
},
"responses": {
"200": {
"description": "成功",
"content": {
"application/json": {
"schema": {
"type": "object",
"properties": {}
},
"examples": {}
}
}
}
}
}
}
},
"components": {
"schemas": {}
}
}