相关博文:

从消费者角度评估RestFul的意义

SpringBoot 构建RestFul API 含单元测试

REST是目前业界相当火热的术语,似乎发布的API不带个REST前缀,你都不好意思和别人打招呼了。 然而大部分号称REST的API实际上并没有达到Richardson成熟度模型的第三个级别:Hypermedia。 而REST的发明者Roy Fielding博士更是直言“Hypermedia作为应用引擎”是REST的前提, 这不是一个可选项,如果没有Hypermedia,那就不是REST。(摘自Infoq对Fielding博士的第二段访谈)

什么是Hypermedia?

  那究竟什么是Hypermedia? 采用Hypermedia的API在响应(response)中除了返回资源(resource)本身外,还会额外返回一组链接(link)。 这组链接描述了对于该资源,消费者(consumer)接下来可以做什么以及怎么做。

举例来说,假设向API发起一次get请求,获取指定订单的资源表述(representation),那么它应该长得像这样:

HTTP/1.1  OK
Server: Apache-Coyote/1.1
Content-Type: application/hal+json;charset=UTF-
Transfer-Encoding: chunked
Date: Fri, Jun :: GMT {
"tracking_id": "",
"status": "WAIT_PAYMENT",
"items": [
{
"name": "potato",
"quantity":
}
],
"_links": {
"self": {
"href": "http://localhost:57900/orders/123456"
},
"cancel": {
"href": "http://localhost:57900/orders/123456"
},
"payment": {
"href": "http://localhost:57900/orders/123456/payments"
}
}
}
  • 理解链接中的“self”的消费者知道使用get方法访问其“href”的uri可以查看该订单的详细信息
  • 理解链接中的“cancel”的消费者知道使用delete方法访问其“href”的uri可以取消该订单
  • 理解链接中的“payment”的消费者知道使用post方法访问其“href”的uri可以为该订单付款

这看起来很有趣,然而这对API的消费者来说有什么好处呢?

不再揣测如何组合使用API

  不知道在你的开发生涯中有没有遇到过这样的事情:

有一天,产品经理跟我说,我们要和某某酒店集团对接,在线销售它们的酒店,这是他们的联系人和详细的API说明文档。 API说明文档真够详细,有好几十页,凭着丰富的行业经验,我知道我需要找到其中的哪些API来实现基本的业务场景。 几天后,我实现了大部分的API集成,现在可以预订酒店了,订单已经在对方的测试环境生成,大功告成。 等等,这个“添加订单财务信息”的API是干嘛的?在和对方的联系人联系后,被告知“没什么用”,好了,真的大功告成了,上线!

两周后,我们的API使用权限被对方关闭了,原因是“所有的订单都没有财务信息,无法确认对账”。
“等等,那谁谁谁不是说这个API没用吗?”
“噢,他已经离职了,你们如果要恢复使用,尽快完成这个API的集成吧”
“我。。。”

当然,这里面还有许多别的因素,但是消费者的开发人员往往很难将业务场景和实现业务场景的API联系起来。 他们常常面对是:

  • 不熟悉的业务场景
  • 一套对单个API的作用描述详细,但缺乏API之间联系的文档。

Hypermedia带来的API自描述特性,使用链接的方式提示接下来做什么和怎么做,正好可以缓解这样的窘境。 如果API服务方可以提供测试环境供消费者测试,那么开发人员可以实际动手探索业务场景的衔接,这时再配合API文档情况就好多了。 回到酒店订单的例子,如果是这样,我可能就不会挨批了:

// 预订后,提示确认订单,那么不熟悉为什么要确认以及不确认的后果的同学就可以想到去问啦
{
"tracking_id": "",
"Hotel": "A ZHAO DAI SUO",
"status": "WAIT_ACKNOWLEDGED",
"_links": {
"self": {
"href": "http://zhaodaisuo.com/orders/123456"
},
"cancel": {
"href": "http://zhaodaisuo.com/orders/123456"
},
"acknowledge": {
"href": "http://zhaodaisuo.com/orders/123456/payments"
}
}
}
// 确认后,提示添加财务信息,不熟悉的同学就可以问了,然而我真的已经问了呀。。。。
{
"tracking_id": "",
"Hotel": "A ZHAO DAI SUO",
"status": "ACKNOWLEDGED",
"_links": {
"self": {
"href": "http://zhaodaisuo.com/orders/123456"
},
"billing": {
"href": "http://zhaodaisuo.com/orders/123456/bill"
}
}
}

从此与API版本说再见

  不知道在你的开发生涯中有没有遇到过这样的事情:

有一天,产品经理跟我说,我们要实现一个新功能blablabla,但是依赖的API版本太老了,这是他们的联系人。
“你好呀,请问我们需要这个信息,但是现在1.3的版本中没有提供,有什么办法吗?”
“你可以升级到2.1的版本就有了”
“那这个版本是不是向后兼容的啊?我们用了其中很多接口哦,我不想其它的集成点出问题”
“那当然,放心吧”

结果当然是个悲伤的故事,“你给我过来,我保证不打死你”。 API的发布方也需要增加新功能,API自身也会随着需求变化,于是产生了版本号

http://www.zhaodaisuo.com/api/v1.2

然而一套API一般会包括多个API,为整套API版本化的粒度太粗了。 一旦消费者希望获得其中某个API的新特性,他/她只能选择全盘升级并仔细测试或者为每个集成点配置单独的uri。 这都不够好,而Hypermedia可以改变这种局面。 由于提供了链接来告诉消费者资源的uri,相对“传统”的REST API,uri变成了一种弱耦合, Hypermedia API只需要公布少量入口uri就可以了。比如,以之前酒店订单的例子,只需发布

http://www.zhaodaisuo.com/orders

后续的确认、财务信息的uri是在实际API调用的时候拿到的,无需事先准备。 消费者和发布者之间的强耦合实际上只剩下入口uri和服务契约(解释资源的含义), 当服务契约新增或是发生破坏性的变化时(例如修改了或删除了参数),只需要在资源表述中增加新的链接。

{
"tracking_id": "",
"Hotel": "A ZHAO DAI SUO",
"status": "ACKNOWLEDGED",
"_links": {
"self": {
"href": "http://zhaodaisuo.com/orders/123456"
},
"billing": {
"href": "http://zhaodaisuo.com/orders/123456/bill"
},
"billing-v1.1": { //billing发生了破坏性变化
"href": "http://zhaodaisuo.com/orders/123456/bill/v1"
},
"coupon": { //新增了优惠券抵用的契约
"href": "http://zhaodaisuo.com/orders/123456/coupon"
}
}
}

这样版本化的粒度就下移到了服务契约的级别,这时消费者就灵活多了,只要按需修改对应的集成点就行了。

彻底与API的内部实现解耦

  不知道在你的开发生涯中有没有遇到过这样的事情:

有一天,产品经理跟我说,我们依赖的一个API发布者通知我,现在判断订单是否能够用优惠券的条件变化,这是他们的联系人。
“你好呀,请问现在订单能否用优惠券的判断条件有什么变化?”
“原来,你们可以通过订单的状态来判断,现在还需要结合订单的来源,参加秒杀活动的订单不能使用优惠券”
“好吧。。。”

于是我在集成代码中做了如下修改:

 if (order.status().equals("WAIT_PAYMENT")) {
if (order.source().equals("miaosha")) {
couponButtonEnabled = false;
} else {
//....
}
} else {
//...
}

好纠结,就不能把API设计成这样吗?

{
"tracking_id": "",
"Hotel": "A ZHAO DAI SUO",
"status": "WAIT_PAYMENT",
"source": "normal",
"_links": {
"coupon": {
"href": "http://zhaodaisuo.com/orders/123456/coupon"
}
}
}
{
"tracking_id": "",
"Hotel": "A ZHAO DAI SUO",
"status": "WAIT_PAYMENT",
"source": "miaosha",
"_links": {} //秒杀来源的订单不返回含优惠券链接的资源表述。
}

这样客户端代码就简单了,依赖于抽象的业务场景,而不是依赖于具体的实现

if (order.containsLink("coupon")) {
couponButtonEnabled = true;
} else {
couponButtonEnabled = false;
}
 
转自: http://hippoom.github.io/blogs/value-of-hypermedia-from-client-perspective.html

从消费者角度评估RestFul的意义的更多相关文章

  1. SpringBoot 构建RestFul API 含单元测试

    相关博文: 从消费者角度评估RestFul的意义 SpringBoot 构建RestFul API 含单元测试 首先,回顾并详细说明一下在快速入门中使用的  @Controller .  @RestC ...

  2. Java实现多线程生产者消费者模型及优化方案

    生产者-消费者模型是进程间通信的重要内容之一.其原理十分简单,但自己用语言实现往往会出现很多的问题,下面我们用一系列代码来展现在编码中容易出现的问题以及最优解决方案. /* 单生产者.单消费者生产烤鸭 ...

  3. 从互联网进化的角度看AI+时代的巨头竞争

    今天几乎所有的互联网公司在谈论和布局人工智能,收购相关企业.人工智能和AI+成为当今科技领域最灸手可热的名词,关于什么是AI+,其概念就是用以表达将"人工智能"作为当前行业科技化发 ...

  4. 第五次作业——python效能分析与几个问题(个人作业)

    第五次作业--效能分析与几个问题(个人作业) 前言 阅读了大家对于本课程的目标和规划之后,想必很多同学都跃跃欲试,迫不及待想要提高自身实践能力,那么就从第一个个人项目开始吧,题目要求见下. 阅读 阅读 ...

  5. 使用MVVM设计模式构建WPF应用程序

    使用MVVM设计模式构建WPF应用程序 本文是翻译大牛Josh Smith的文章,WPF Apps With The Model-View-ViewModel Design Pattern,译者水平有 ...

  6. MapReduce剖析笔记之八: Map输出数据的处理类MapOutputBuffer分析

    在上一节我们分析了Child子进程启动,处理Map.Reduce任务的主要过程,但对于一些细节没有分析,这一节主要对MapOutputBuffer这个关键类进行分析. MapOutputBuffer顾 ...

  7. 苹果5S指纹扫描识别传感器Touch ID有利于iPhone的安全性

    iPhone5S新增的指纹扫描识别传感器 Touch ID,黑客花了大量的时间表明指纹验证是可以被破解的.即使它可能被黑客攻击,对iPhone5S的安全性而言,仍然具有极大的好处. 为什么一个容易被破 ...

  8. ITU-T Technical Paper: NP, QoS 和 QoE的框架以及它们的区别

    本文翻译自ITU-T的Technical Paper:<How to increase QoS/QoE of IP-based platform(s) to regionally agreed ...

  9. Kafka 0.11.0.0 实现 producer的Exactly-once 语义(中文)

    很高兴地告诉大家,具备新的里程碑意义的功能的Kafka 0.11.x版本(对应 Confluent Platform 3.3)已经release,该版本引入了exactly-once语义,本文阐述的内 ...

随机推荐

  1. windows32位系统 安装MongoDB

    今天在win7 32位系统下安装 MongoDB 时,遇到了一堆坑,特此笔记. 一.下载MongoDB 打开官网下载地址:https://www.mongodb.com/download-center ...

  2. Beta冲刺——day5

    Beta冲刺--day5 作业链接 Beta冲刺随笔集 github地址 团队成员 031602636 许舒玲(队长) 031602237 吴杰婷 031602220 雷博浩 031602134 王龙 ...

  3. WBS功能分解及甘特图

    产品 一级子功能 二级子功能 三级子功能 时间(小时)  食物链教学工具 属性面板 功能按键 选择环境 1       自定义生物 2       生物连线与删除 5       显示食物链 1   ...

  4. pgm1

    很遗憾前面只看过 Michael Jordan 写的一部分,这次打算把 Daphne Koller 和 Nir Friedman 合著的 Probabilistic Graphical Models: ...

  5. Pathwalks CodeForces - 960F(主席树 || 树状数组)

    题意: 求树上最长上升路径 解析: 树状数组版: 998ms edge[u][w] 代表以u为一条路的终点的小于w的最长路径的路的条数 · 那么edge[v][w] = max(edge[u][w-1 ...

  6. Java 8 中 Date与LocalDateTime、LocalDate、LocalTime互转

    Java 8中 java.util.Date 类新增了两个方法,分别是from(Instant instant)和toInstant()方法 // Obtains an instance of Dat ...

  7. Linux掉电处理

    在嵌入式设备中,掉电处理一直是一项比较麻烦的工作,在具有Linux系统的设备中,系统的种种数据的处理更是增加掉电处理的难度.现在做以下几点总结,再遇到类似问题可以做个参考. 1,系统启动的处理 在系统 ...

  8. python学习(28) 浅谈可变对象的单例模式设计

    python开发,有时候需要设计单例模式保证操作的唯一性和安全性.理论上python语言底层实现和C/C++不同,python采取的是引用模式,当一个对象是可变对象,对其修改不会更改引用的指向,当一个 ...

  9. linux basic ------ 多命令执行

    当我们需要一次执行多个命令的时候,命令之间需要用连接符连接,不同的连接符有不同的效果.下面我们总结一下,加以区分. (1)  ; 分号,没有任何逻辑关系的连接符.当多个命令用分号连接时,各命令之间的执 ...

  10. linux命令总结之echo命令

    echo是一种最常用的与广泛使用的内置于Linux的bash和C shell的命令,通常用在脚本语言和批处理文件中来在标准输出或者文件中显示一行文本或者字符串. echo命令的语法是: echo [选 ...