【RESTful】深入理解 RESTful Api 架构

非原创,搬运地址: http://mengkang.net/623.html

一些常见的误解

不要以为 RESTful Api 就是设计得像便于 SEO 的伪静态,例如一个 ApiURL 类似于 http://xxx.com/blog/1 ,我们可以通过浏览器访问该 URL 而读取文章,但是这并不代表着它就是 RESTful Api
也不要认为URL 里有 queryString 就不是 RESTful Api ,例如 http://xxx.com/users/?page=10&page_size=30
更不要认为 HTTP + JSON 就是 RESTful ApI 了。

REST 和 HTTP/1.1

Roy FieldingApache HTTP 服务器的核心开发者,Apache 软件基金会的合作创始人,HTTP/1.1 协议专家组的负责人)总结了一套理论框架,然后使用这套理论框架中的指导原则,来指导HTTP/1.1协议的设计方向。后来他在其的博士学位论文 Architectural Styles and the Design of Network-based Software Architectures 中更为系统、严谨地阐述了这套理论框架,并且使用这套理论框架推导出了一种新的架构风格,并且为这种架构风格取了一个令人轻松愉快的名字 REST
想必通过这个你已经明白了 RESTHTTP/1.1 的密不可分的关系了吧。HTTP/1.1 的很多特性就是 REST 的特性,比如连接的无状态性。还有后面说的 REST 五大特性都和 HTTP/1.1 协议密不可分。

RESTful Api 与 SOAP Web API 在 URL 形式上的对比

从设计一个删除评论的 api 说起
我们可以这样设计成类似于:

http://mengkang.net/?method=comment.del&id=x ①
http://mengkang.net/comment/del/id/x ②

或者其他形式的 url。而 RESTful Api 则是:

[DELETE] http://mengkang.net/comments/1 ③

我们对比可以发现①和② URL 中,都有del的动作指示。

SOAP Web API采用RPC风格,它采用面向功能的架构,所以我们在设计SOAP Web API的时候首相考虑的是应高提供怎样的功能(或者操作)。
RESTful Api 是面向资源的架构。是查询、新增、修改、删除,都该资源无关。

所以我们在③ URL 中没有看到del的关键字,对比②和③最为明显。

进一步认识 RESTful Api

我们知道 URL 全称为“统一资源定位符(Uniform Resource Locator)”,用于描述 Web 资源所在的位置。RESTful Api 是以 HTTP 协议为强烈依托的,将类似于①和②这种以功能为主导的URL风格舍弃,还原 URL 的本质,它的宗旨就是一个 URL 就应该是一个资源,不能包含任何动作。

[POST]     http://mengkang.net/users   // 新增
[GET]      http://mengkang.net/users/1 // 查询
[PATCH]    http://mengkang.net/users/1 // 更新
[PUT]      http://mengkang.net/users/1 // 覆盖,全部更新
[DELETE]   http://mengkang.net/users/1 // 删除

PUTPATCH的功能都可以代表更新,但略有不同,PUT大多时候表示更新该资源的全部信息,而PATCH则更新部分信息。

PUTPOST又一些功能的重叠,都可以是新建一个资源,POST时,新建资源的地址是由服务器返回给客户端的。也就是说客户端在发送POST请求资源之前还无法预知该资源的地址,这在我们的 Api 开发中非常常见,新建一个帖子,新建一条评论,都如此。

而资源的地址是客户端预先知道的情况则比较少。也有人如此设计而使用PUT,比如需要对 github 上的某人的项目 star ,则可能会设计成:http://github.com/xxx/zhoumengkang/projectname/star 这里把“对这个项目已经点赞过”看成了一个资源,那么就可以用PUT,因为要删除这个资源时,还是访问这个 URL

关于这些功能上有交集的地方,可能后面会有一些更加标准的规范或者协议吧。

最后说下HEADOPTIONSHEAD请求的是资源的元数据,比如一张照片,的元数据则可能包含了,照片拍摄的设备,地点,时间等。服务器一般将对应资源的元数据置于响应的报头集合返回给客户端,这样的响应一般不具有主体部分。OPTIONS则是发送一种“探测”请求以确定针对某个目标地址的请求必须具有怎样的约束(比如应该采用怎样的HTTP方法以及自定义的请求报头),然后根据其约束发送真正的请求。

关于过滤筛选,排序和 tokenquery string 是支持使用,和唯一资源的概念并不冲突。

我所理解的无状态性和唯一资源对 Api Url 的设计指导

举一个实际的例子,用户黑名单的 CURD。我们可是设计成

[GET]      /blacklist
[PUT]      /blacklist/{id} #把 id 加入到当前授权用户的黑名单中
[DELETE]   /blacklist/{id}

如上的 api 设计中,首先如上的 url 设计中,通过授权码中获取登录用户 uid。则是有状态的了,其次因为所有的用户访问的资源都是同一个地址,这与唯一资源的概念是相违背的。如果是无状态的,那么就与唯一资源的概念相吻合了。(这只是 restful 所建议的,实际是否需要完全遵守视情况而定)

[GET]      /user/{uid}/blacklist
[PUT]      /user/{uid}/blacklist/{id}  #把 id 加入到当前授权用户的黑名单中
[DELETE]   /user/{uid}/blacklist/{id}
REST的五大特性
  1. 资源(Resource)
  2. 资源的表述(Representation)
  3. 状态转移(State Transfer)
  4. 统一接口(Uniform Interface)
  5. 超文本驱动(Hypertext Driven)
1. 资源

关于资源的理解在http://mengkang.net/620.html里已经说了很多了。

2. 资源的表述

实际是资源暴露出来的展现形式,拿[GET]访问的请求来说就是通过浏览器访问某个地址之后所得到的内容。

3. 状态转移

我所理解的,将客户端对资源操作的状态通过 Api 的请求转移到服务端。比如客户端需要删除某个资源,请求完,服务器端该资源的状态也是删除的了。

4. 统一接口

REST要求,必须通过统一的接口来对资源执行各种操作。对于每个资源只能执行一组有限的操作。以HTTP/1.1协议为例,HTTP/1.1协议定义了一个操作资源的统一接口,主要包括以下内容:

  1. 7个HTTP方法:GET/POST/PUT/DELETE/PATCH/HEAD/OPTIONS
  2. HTTP头信息(可自定义)
  3. HTTP响应状态代码(可自定义)
  4. 一套标准的内容协商机制(即接口内容输出之后,各个字段各个对象如何解析的定义)
  5. 一套标准的缓存机制
  6. 一套标准的客户端身份认证机制
5. 超文本驱动

就像网页里的超链接,得到当前接口的返回内容,app 布局上对应的各个按钮下一步可能会请求 api 需要在当前 api 予以给出。

举个例子 https://developer.github.com/v3/git/commits/#get-a-commit 返回的内容为:

{
  "sha": "7638417db6d59f3c431d3e1f261cc637155684cd",
  "url": "https://api.github.com/repos/octocat/Hello-World/git/commits/7638417db6d59f3c431d3e1f261cc637155684cd",
  "author": {
    "date": "2014-11-07T22:01:45Z",
    "name": "Scott Chacon",
    "email": "schacon@gmail.com"
  },
  "committer": {
    "date": "2014-11-07T22:01:45Z",
    "name": "Scott Chacon",
    "email": "schacon@gmail.com"
  },
  "message": "added readme, because im a good github citizen\n",
  "tree": {
    "url": "https://api.github.com/repos/octocat/Hello-World/git/trees/691272480426f78a0138979dd3ce63b77f706feb",
    "sha": "691272480426f78a0138979dd3ce63b77f706feb"
  },
  "parents": [
    {
      "url": "https://api.github.com/repos/octocat/Hello-World/git/commits/1acc419d4d6a9ce985db7be48c6349a0475975b5",
      "sha": "1acc419d4d6a9ce985db7be48c6349a0475975b5"
    }
  ]
}

里面的parents节点里有一个url字段,而对应的客户端上有一个查看其父节点的按钮,点击那个按钮,就会向这个url对应的api发送请求了。
下一步操作的执行,不是客户端需要自身关心的,而是api返回的文档内容驱动的。

个人觉得,这些标准实际在我们开发的时候无形中就实现并遵守了,就想数据库设计范式一样,不遵守,没法玩了。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值