理解 RESTful

前言

近十年,前端高速发展,整个互联网应用经历了从轻客户端到重客户端的变化,随着前端规模越来越大,交互越来越复杂,前后端分离的设计开始流行。

移动互联网时代的到来,前端开始泛指各种终端和 web 前端,服务端为多终端提供服务已然成常态。

前后端分离后,前端和后端通过双方协商好的 API 进行交互,所以设计一套友好的 API 尤其重要。

RESTful 就是目前最流行的前后端交互 API 设计范式。

RESTful 释义

顾名思义,先看一下 RESTful 的单词拆解:

RESTful = Resources + Representation + State + Transfer + ful

我的理解是, RESTful 是指具有 资源表现层状态转换 的架构设计。

资源(Resources)

资源,是指服务端向外提供的服务实体。

资源是一个抽象的概念,可以是应用程序对象、数据库记录、算法等等。每一个资源用一个 URI 来唯一标识,客户端通过这个标识对资源进行访问、操作或请求服务。

表现层(Representation)

表现层,是指资源的表现形式。

一个资源可以有多种表现形式。例如,文本数据可以用 XML 格式、JSON 格式表现,甚至可以用二进制格式表示;图片可以用 JPG 格式表现,也可以用 PNG 格式表现。

状态(State)

状态,是指资源的某种状态。

一个资源可能有多种状态。例如,资源空闲的、占用的、共享的、存在的、不存在的等。

转化(Transfer)

转化,是指对资源的操作行为。

转化的行为包括两种:

  • 资源表现层的转化
  • 资源状态的转化

小结

RESTful 是一种抽象的、与具体程序语言和网络协议无关的网络服务系统的架构样式。

它把在服务器端的数据和功能设计成各种的资源,并且通过 URI 定位资源、转化资源表现和状态的一种服务架构设计。

RESTful WEB

REST 指的是一组架构约束条件和原则。满足这些约束条件和原则的应用程序或设计就是 RESTful。

目前 RESTful 应用最多的是在 web 服务,在 RESTful Web 服务中,每个资源都有一个地址(URL)。资源是方法调用的目标,方法列表对所有资源都是一样的。这些方法是 Http 标准方法,如 HTTP GET、POST、PUT、DELETE 等。

下面来描述一下 RESTful Web API 的一些设计原则。

通信协议

数据传输,安全第一。

HTTPS 协议要优于 HTTP 协议。

API 的根 URL

API 的根 URL 很重要,它应该设计成可以轻松区别于其他非 API 的 URL。

最好的做法是将 API 部署在专有的域名下。

https://api.example.com

退而求其次,分配一个域名一级目录给 API。

https://example.org/api/

版本信息

为了 API 演变和兼容,为 API 加入版本信息是明智之举。

应该为 API 设定版本号,并把版本号信息加入到 API 的根 URL 的下一级路径。

https://api.example.com/v1/

https://example.org/api/v1/

URL 末端

API 的根 URL + 版本信息 + URL 末端 = 资源位置

URL 末端最终明确指定了一个资源或一种资源的集合。URL 末端中不应该出现动词,只能是名词。

举个例子,有一组 API 提供公司员工的信息。

那么,获取所有员工信息,末端应该设计成 /employees, 即:

https://api.example.com/v1/employees

获取员工编号 007 员工的信息,末端应该设计成 /employees/007, 即:

https://api.example.com/v1/employees/007

操作资源

资源的操作无外乎增、删、改、查,对应即是 HTTP 的各个操作方法。

HTTP的操作方法在 RESTful 中有各自的语义,理解它们的语义至为重要。

方法 语义 例子 说明
GET 选择、获取 GET /employees/007 获取编号 007 员工的信息
POST 新建 POST /employees 新建一个员工的信息
PUT 更新(完整) PUT /employees/007 更新编号 007 员工的全部信息,客户端提供该员工的全部信息
PATCH 更新(局部) PATCH /employees/007 更新编号 007 员工的部分信息,客户端只提供该员工的部分信息
DELETE 删除 DELETE /employees/007 删除编号 007 员工的信息
HEAD 获取(元数据) HEAD /employees/007 获取编号 007 员工的元数据,如员工数据的哈希值或最后修改时间
OPTIONS 获取(权限信息) OPTIONS /employees/007 获取客户端能编号 007 员工进行哪些操作,即操作的权限

本章中,自此以上讲述的都是如何描述"资源(Resources)",而这小节讲述的是"转化(Transfer)",转化资源的状态。

另外,转化还包括资源的"表现层(Representation)",它是通过 HTTP 头部 Content Type 指定的。

过滤资源

如果资源很多,通常你不会希望服务端一次性全部的数据,API 应该提供过滤资源的能力。

这里资源过滤依靠 URL 的查询参数,通常放置在 GET 请求的URL中。

  • ?limit=10: 限制返回的资源数量
  • ?offset=10: 指定返回的资源的开始位置
  • ?sortby=name&order=asc: 对资源按特定属性进行排序

状态码

状态码用以反映资源的操作结果或所处状态。

HTTP 状态码本身是具有语义的,这刚好配对了 RESTful 的状态。

以下列举常用的状态码:

状态码 语义 HTTP 方法 说明
200 OK GET 成功返回用户请求的数据
201 Created POST/PUT/PATCH 用户新建或修改资源成功
202 Accepted * 成功发起一个异步任务
204 No Content DELETE 删除资源成功
400 INVALID REQUEST POST/PUT/PATCH 请求有错误,服务端没有对资源进行任何操作
401 Unauthorized * 表示用户没有权限(令牌、用户名、密码错误)
403 Forbidden * 表示用户得到授权(与401错误相对),但是访问是被禁止的
404 NOT FOUND * 请求对应的资源不存在
406 Not Acceptable GET 用户请求的格式不可得(比如用户请求JSON格式,但是只有XML格式), 即未支持的表现层
410 Gone GET 用户请求的资源被永久删除,且不会再得到的
422 Unprocesable entity POST/PUT/PATCH 当创建一个对象时,发生一个验证错误

错误处理

如果状态码是4xx,就应该向用户返回出错信息。一般来说,返回的信息中将error作为键名,出错信息作为键值。

{
    error: "param error"
}

返回结果

对于不同操作方法和操作对象(集合或个体),服务器返回的结果应该符合以下规范。

示例 返回
GET /collection 返回资源对象的列表(数组)
GET /collection/resource 返回单个资源对象
POST /collection 返回新生成的资源对象
PUT /collection/resource 返回完整的资源对象
PATCH /collection/resource 返回完整的资源对象
DELETE /collection/resource 返回一个空文档

另外,返回的数据格式(Representation)应该尽量使用JSON。

最后

在 RESTful WEB 的实现中,使用 URL 定位资源,HTTP 方法操作资源,使得语义明确、结构清晰、易于理解和扩展。

RESTful 的这些优势,使得它成为了目前最流行的一种互联网服务架构。

附录

该文主要参考:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容