1
0
mirror of https://github.com/Snailclimb/JavaGuide synced 2025-06-16 18:10:13 +08:00
2020-03-27 09:14:52 +08:00

141 lines
9.3 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

![bd4442aed16acafc54c7943d34abff0edadfa74c](https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-11/bd4442aed16acafc54c7943d34abff0edadfa74c.png)
大家好我是 Guide 哥!这是我的第 **210** 篇优质原创!这篇文章主要分享了后端程序员必备的 RestFul API 相关的知识。
RESTful API 是每个程序员都应该了解并掌握的基本知识,我们在开发过程中设计 API 的时候也应该至少要满足 RESTful API 的最基本的要求(比如接口中尽量使用名词,使用 POST 请求创建资源DELETE 请求删除资源等等,示例:`GET /notes/id`:获取某个指定 id 的笔记的信息)。
如果你看 RESTful API 相关的文章的话一般都比较晦涩难懂,包括我下面的文章也会提到一些概念性的东西。但是,实际上我们平时开发用到的 RESTful API 的知识非常简单也很容易概括!举个例子,如果我给你下面两个 url 你是不是立马能知道它们是干什么的!这就是 RESTful API 的强大之处!
**RESTful API 可以你看到 url + http method 就知道这个 url 是干什么的,让你看到了 http 状态码status code就知道请求结果如何。**
```
GET /classes列出所有班级
POST /classes新建一个班级
```
下面的内容只是介绍了我觉得关于 RESTful API 比较重要的一些东西,欢迎补充。
### 一、重要概念
REST,即 **REpresentational State Transfer** 的缩写。这个词组的翻译过来就是"表现层状态转化"。这样理解起来甚是晦涩,实际上 REST 的全称是 **Resource Representational State Transfe** ,直白地翻译过来就是 **“资源”在网络传输中以某种“表现形式”进行“状态转移”** 。如果还是不能继续理解,请继续往下看,相信下面的讲解一定能让你理解到底啥是 REST 。
我们分别对上面涉及到的概念进行解读,以便加深理解,不过实际上你不需要搞懂下面这些概念,也能看懂我下一部分要介绍到的内容。不过,为了更好地能跟别人扯扯 “RESTful API”我建议你还是要好好理解一下
- **资源Resource** :我们可以把真实的对象数据称为资源。一个资源既可以是一个集合,也可以是单个个体。比如我们的班级 classes 是代表一个集合形式的资源,而特定的 class 代表单个个体资源。每一种资源都有特定的 URI统一资源定位符与之对应如果我们需要获取这个资源访问这个 URI 就可以了,比如获取特定的班级:`/class/12`。另外,资源也可以包含子资源,比如 `/classes/classId/teachers`:列出某个指定班级的所有老师的信息
- **表现形式Representational**"资源"是一种信息实体,它可以有多种外在表现形式。我们把"资源"具体呈现出来的形式比如 jsonxmlimage,txt 等等叫做它的"表现层/表现形式"。
- **状态转移State Transfer** :大家第一眼看到这个词语一定会很懵逼?内心 BB这尼玛是啥啊 大白话来说 REST 中的状态转移更多地描述的服务器端资源的状态,比如你通过增删改查(通过 HTTP 动词实现引起资源状态的改变。ps:互联网通信协议 HTTP 协议,是一个无状态协议,所有的资源状态都保存在服务器端。
综合上面的解释,我们总结一下什么是 RESTful 架构:
1. 每一个 URI 代表一种资源;
2. 客户端和服务器之间,传递这种资源的某种表现形式比如 jsonxmlimage,txt 等等;
3. 客户端通过特定的 HTTP 动词,对服务器端资源进行操作,实现"表现层状态转化"。
### 二、REST 接口规范
#### 1、动作
- GET :请求从服务器获取特定资源。举个例子:`GET /classes`(获取所有班级)
- POST :在服务器上创建一个新的资源。举个例子:`POST /classes`(创建班级)
- PUT :更新服务器上的资源(客户端提供更新后的整个资源)。举个例子:`PUT /classes/12`(更新编号为 12 的班级)
- DELETE :从服务器删除特定的资源。举个例子:`DELETE /classes/12`(删除编号为 12 的班级)
- PATCH :更新服务器上的资源(客户端提供更改的属性,可以看做作是部分更新),使用的比较少,这里就不举例子了。
#### 2、路径接口命名
路径又称"终点"endpoint表示 API 的具体网址。实际开发中常见的规范如下:
1. **网址中不能有动词只能有名词API 中的名词也应该使用复数。** 因为 REST 中的资源往往和数据库中的表对应,而数据库中的表都是同种记录的"集合"collection。**如果 API 调用并不涉及资源(如计算,翻译等操作)的话,可以用动词。** 比如:`GET /calculate?param1=11&param2=33`
2. 不用大写字母,建议用中杠 - 不用下杠 \_ 比如邀请码写成 `invitation-code`而不是 ~~invitation_code~~
Talk is cheap来举个实际的例子来说明一下吧现在有这样一个 API 提供班级class的信息还包括班级中的学生和教师的信息则它的路径应该设计成下面这样。
**接口尽量使用名词,禁止使用动词。** 下面是一些例子:
```
GET /classes列出所有班级
POST /classes新建一个班级
GET /classes/classId获取某个指定班级的信息
PUT /classes/classId更新某个指定班级的信息一般倾向整体更新
PATCH /classes/classId更新某个指定班级的信息一般倾向部分更新
DELETE /classes/classId删除某个班级
GET /classes/classId/teachers列出某个指定班级的所有老师的信息
GET /classes/classId/students列出某个指定班级的所有学生的信息
DELETE classes/classId/teachers/ID删除某个指定班级下的指定的老师的信息
```
反例:
```
/getAllclasses
/createNewclass
/deleteAllActiveclasses
```
理清资源的层次结构,比如业务针对的范围是学校,那么学校会是一级资源:`/schools`,老师: `/schools/teachers`,学生: `/schools/students` 就是二级资源。
#### 3、过滤信息Filtering
如果我们在查询的时候需要添加特定条件的话,建议使用 url 参数的形式。比如我们要查询 state 状态为 active 并且 name 为 guidegege 的班级:
```
GET /classes?state=active&name=guidegege
```
比如我们要实现分页查询:
```
GET /classes?page=1&size=10 //指定第1页每页10个数据
```
#### 4、状态码Status Codes
**状态码范围:**
| 2xx成功 | 3xx重定向 | 4xx客户端错误 | 5xx服务器错误 |
| --------- | -------------- | ---------------- | --------------- |
| 200 成功 | 301 永久重定向 | 400 错误请求 | 500 服务器错误 |
| 201 创建 | 304 资源未修改 | 401 未授权 | 502 网关错误 |
| | | 403 禁止访问 | 504 网关超时 |
| | | 404 未找到 | |
| | | 405 请求方法不对 | |
### 三 HATEOAS
> **RESTful 的极致是 hateoas ,但是这个基本不会在实际项目中用到。**
上面是 RESTful API 最基本的东西也是我们平时开发过程中最容易实践到的。实际上RESTful API 最好做到 Hypermedia即返回结果中提供链接连向其他 API 方法,使得用户不查文档,也知道下一步应该做什么。
比如,当用户向 api.example.com 的根目录发出请求,会得到这样一个文档。
```javascript
{"link": {
"rel": "collection https://www.example.com/classes",
"href": "https://api.example.com/classes",
"title": "List of classes",
"type": "application/vnd.yourformat+json"
}}
```
上面代码表示,文档中有一个 link 属性,用户读取这个属性就知道下一步该调用什么 API 了。rel 表示这个 API 与当前网址的关系collection 关系,并给出该 collection 的网址href 表示 API 的路径title 表示 API 的标题type 表示返回类型 Hypermedia API 的设计被称为[HATEOAS](http://en.wikipedia.org/wiki/HATEOAS)。
在 Spring 中有一个叫做 HATEOAS 的 API 库,通过它我们可以更轻松的创建除符合 HATEOAS 设计的 API。
### 文章推荐
**RESTful API 介绍:**
- [RESTful API Tutorial](https://RESTfulapi.net/)
- [RESTful API 最佳指南](http://www.ruanyifeng.com/blog/2014/05/RESTful_api.html)(阮一峰,这篇文章大部分内容来源)
- [[译] RESTful API 设计最佳实践](https://juejin.im/entry/59e460c951882542f578f2f0)
- [那些年,我们一起误解过的 REST](https://segmentfault.com/a/1190000016313947)
- [Testing RESTful Services in Java: Best Practices](https://phauer.com/2016/testing-RESTful-services-java-best-practices/)
**Spring 中使用 HATEOAS**
- [在 Spring Boot 中使用 HATEOAS](a)
- [Building REST services with Spring](https://spring.io/guides/tutorials/classmarks/) (Spring 官网 )
- [An Intro to Spring HATEOAS](https://www.baeldung.com/spring-hateoas-tutorial) by [baeldung](https://www.baeldung.com/author/baeldung/)
- [spring-hateoas-examples](https://github.com/spring-projects/spring-hateoas-examples/tree/master/hypermedia)
- [Spring HATEOAS](https://spring.io/projects/spring-hateoas#learn) (Spring 官网 )