定义资源

在Fielding的论文中 ,资源被描述为:

“可以命名的任何信息”……“文档或图像,临时服务(例如,“洛杉矶今天的天气”),其他资源的集合,非虚拟对象(例如,人) 等等。 换句话说,任何可能成为作者超文本 引用 目标的概念都 必须符合资源的定义。 资源是 到一组实体 的概念性映射 ,而不是在任何特定 时间 点对应于该映射的实体 。”

定义资源既是科学也是艺术 。 它需要领域知识和API体系结构技能。 下面详细介绍的以下几点用作清单,可以帮助您确定资源。

资源必须包含业务说明

  • 商业描述应为简单散文中的3-4个句子,以说明资源是什么。
  • 对您的系统有一定了解的开发人员应该能够理解该描述
  • 资源的任何警告均应明确

资源应单独使用

这类似于定义微服务边界的准则,在这种情况下,应将微服务视为自身有用。 同样,资源应单独使用。

例如,代替:

 /street-address/{id}  RESPONSE  { "street1" : "String" , "street2" : "String"  } 

 /address-extra/{id}  RESPONSE  { "city" : "String" , "country" : "String"  } 

它应该是:

 /address/{id}  RESPONSE  { "street1" : "String" , "street2" : "String" , "city" : "String" , "country" : "String"  } 

如果资源本身没有用,并且总是需要后续请求,则意味着代码将不可避免地变得更加复杂,并且第二个请求将对性能造成影响

使用适当的名词

最好使用简单名词而不是复合名词。 例如,
地址优于AddressInfoAddressDetail 。 这是一条一般规则,总会有例外

如果使用多个资源表示同一数据的不同视图,例如: AddressAddressDetail ,则使用简单名词,例如
地址第一。 然后,如果第二种表示形式更详细地使用
ResourceNameDetail,或者如果不够详细,请使用ResourceNameSummary 。 例如,假设需要引入一个地址类型资源:

  1. 地址首先介绍
  2. 如果需要更详细的地址后续视图,则新资源应称为AddressDetail
  3. 如果需要的后继地址视图不太详细,则新资源应称为AddressSummary

如果仅在READ中使用它,它是否需要成为资源?

如果仅在读取请求中使用过资源,而从未在写入创建,部分更新,完全更新,删除等 )请求中使用过资源,则是否需要将其定义为具有自己的URI的资源是可疑的。 可以将其添加到父负载中,如果担心负载会变得太复杂,则父可以仅提供一个稀疏查询-客户端可以根据API请求确定要返回的内容。

资源应符合统一接口

统一的接口是良好API设计的重要组成部分。 如果创建,读取,更新,删除等操作以一致的方式进行,则意味着代码更加一致,可重用且更易于维护。

这表示:

 GET /addresses/{id} 

 GET /addresses 

必须返回相同的地址数据结构来表示一个地址。

 GET /addresses/{id}  RESPONSE  { "id" : "546" , "street1" : "String" , "street2" : "String" , "city" : "String" , "country" : "String"  } 

 GET /addresses  RESPONSE  { "elements" : [ { "id" : "546" , "street1" : "String" , "street2" : "String" , "city" : "String" , "country" : "String" }, ... ]  } 

同样,对于写入有效负载,数据结构应该相同。 因此,更改street1的部分更新将是:

 POST /addresses/{id}/edit  REQUEST  { "street1" : "Walkview"  }  RESPONSE  { "id" : "546" , "street1" : "Walkview" , "street2" : "Meadowbrook" , "city" : "Dublin" , "country" : "Ireland"  } 

而不是像

 POST /addresses/{id}  REQUEST  { "newStreet1Value" : "Walkview"  } 

从资源的角度来看,数据结构必须一致。 不同的数据结构意味着不同的资源,应使用不同的名称命名并具有自己的路径。

不要暴露一切

如果您的数据库模型非常复杂,则不需要在API级别公开所有属性。 有些字段只能保留用于后台处理,而不能显示在UI上。 此类属性永远不应包含在JSON API中。

在将属性添加到JSON资源时,请考虑:

  • API中应仅公开您确定客户端感兴趣的字段
  • 如果不确定,请忽略该属性。 稍后添加属性,然后删除已经公开的属性的风险要低得多。

API模型不应盲目地反映数据库关系模型或OO模型

在数据库建模方法中,使用规范化数据或折叠继承层次结构。 在面向对象的设计中,诸如多态,继承层次结构等技术被用于促进诸如代码重用之类的事情并减少耦合。

资源建模不必遵循这些技术。 API的使用者不关心数据是全部放在一个表中还是在多个表中进行了规范化。 通常,API以易于使用的格式返回数据,并且在客户端变得有用之前不需要客户端进行太多其他映射。

使用分层数据避免重复

与诸如CSV之类的平面格式相比,分层数据的优点之一是它提供了一种避免重复的机制。 例如,考虑一个数据结构,其中包含人员列表以及他们所在的团队。在CSV中,这是:

 team, firstname, lastname  Liverpool, Mo, Salah  Liverpool, Andy, Roberston 

在JSON中,可能是:

 { "team" : "Liverpool" , "players" : [ { "firstName" : "Mo" , "lastName" : "Salah" }, { "firstName" : "Andy" , "lastName" : "Roberston" }, ... ]  } 

使用分层数据使上下文清晰

分层数据的另一个优点是它有助于提供上下文。 要了解平面数据结构,您需要了解生成数据结构的查询是什么,以了解其含义。 例如,考虑一堆包含日期范围的行。

 name, fromDate, toDate, holidays  Tony, 2018 - 01 - 01 , 2018 - 02 - 02 , true  Tony, 2018 - 02 - 03 , 2018 - 03 - 01 , false 

您可以假设当Tony休假时会有新的一行。 但是如果还有另一列怎么办

 name, fromDate, toDate, holidays, sick,  Tony, 2018 - 01 - 01 , 2018 - 02 - 02 , true , false  Tony, 2018 - 02 - 03 , 2018 - 03 - 01 , false , true 

日期范围是否对应于假期,疾病或两者兼而有之?

如果我们能获得更多数据,也许会更清楚……

 name, fromDate, toDate, holidays, sick,  Tony, 2018 - 01 - 01 , 2018 - 02 - 02 , true , false  Tony, 2018 - 02 - 03 , 2018 - 03 - 01 , false , true  Tony, 2018 - 03 - 02 , 2018 - 04 - 01 , false , false 

现在看来,日期范围所对应的是一种病,这只是一个巧合,它排列了一个假期。 但是,当我们获得更多数据时,此理论将失败:

 name, fromDate, toDate, holidays, sick,  Tony, 2018 - 01 - 01 , 2018 - 02 - 02 , true , false  Tony, 2018 - 02 - 03 , 2018 - 03 - 01 , false , true  Tony, 2018 - 03 - 02 , 2018 - 04 - 01 , false , false  Tony, 2018 - 04 - 02 , 2018 - 05 - 01 , true , false 

平面数据结构的问题在于,只能使数据自我描述。 如果没有任何信息,它将变得更加复杂。 例如:

 name, fromDate, toDate, holidays, sick,  Tony, 2018 - 01 - 01 , 2018 - 02 - 02 , true , false  Tony, 2018 - 02 - 03 , 2018 - 03 - 01 , false , true  Tony, 2018 - 03 - 02 , 2018 - 04 - 01 , false , false  Tony, 2018 - 04 - 02 , 2018 - 05 - 01 , true , false  Tony, 2018 - 05 - 02 , 2018 - 06 - 01 , null , false  Tony, 2018 - 06 - 02 , 2018 - 07 - 01 , null , false  Tony, 2018 - 07 - 02 , 2018 - 07 - 08 , true , false  Tony, 2018 - 07 - 08 , 2018 - 07 - 09 , true , null 

不可避免的是,处理该数据将是错误的。 我们可以用以下分层格式表示相同的数据:

 { "name" : "tony" , "holidays" : [ { "fromDate" : "fromDate" "2018-01-01" , "toDate" : "2018-02-02" }, { "fromDate" : "fromDate" "2018-04-02" , "toDate" : "2018-05-01" }, { "fromDate" : "2018-07-02" , "toDate" : "2018-07-09" } ], "sick" : [ { "fromDate" : "2018-02-03" , "toDate" : "2018-03-01" } ]  } 

现在,数据更加自我描述。 很清楚,日期范围是假期还是病假。

资源关系

资源本身只能描述自己。 资源模型描述资源之间的关系。 这将表明:

  • 资源之间的依赖关系。 存在特定资源需要哪些资源,或者当特定资源发生更改时会影响哪些资源:更新或删除。
  • 数据导航–在大域模型中,如果向模型的使用者提供导航和方向感,则更容易理解和遵循。 尤其是何时进行导航(资源松散连接)与向下导航(资源牢固连接)可以区分开

资源不仅应该考虑实现HATEOAS的超媒体链接; 当资源使用超媒体链接描述它们所链接的内容时,它是表达资源模型的一种非常强大的机制。 优势包括:

  • 它将大型域模型分成更易于管理的部分。 通常,用户只对模型的特定部分感兴趣。 当“资源”自己描述自己的关系时,这意味着将一个大型的复杂模型分解为易于消化的部分,并且用户可以更快地获取所需的信息。
  • 资源模型是自我描述的,并与代码保持同步。 一切都在同一地点。

明确父母与子女的关系

子级自我描述了父级URL层次名称间距。 父资源具有一种或多种类型的子代,应通过提供指向子代的链接来阐明这一点。 例如,如果一个团队有一个玩家。 团队有效负载应对此进行明确说明。

 REQUEST  https: //api.server.com/teams/4676  RESPONSE  { "id" : "34533" , ..., "_links" : { "self" : " https://api.server.com/teams/4676 " , "players" : " https://api.server.com/teams/4676/players " }  } 

明确对等关系

这与上面的类似,只不过它用于存在于不同层次名称空间中的资源。 因此,例如,假设团队在部门1中。应该在团队的部门属性中包含一个链接。

 REQUEST  https: //api.server.com/teams/4676  RESPONSE  { "id" : "34533" , "division" : { "name" : "Division 1" , "_links" : { "self" : " https://api.server.com/divisions/1 " } }, ..., "_links" : { "self" : " https://api.server.com/teams/4676 " , "players" : " https://api.server.com/teams/4676/players " }  } 

明确链接到其他表示形式

如果将数据建模为具有多个代表数据不同表示形式的资源,则这些资源还应包括彼此的链接。

 REQUEST  https: //api.server.com/teams/4676  RESPONSE  { "id" : "34533" , "division" : { "name" : "Division 1" , "_links" : { "self" : " https://api.server.com/divisions/1 " } }, ..., "_links" : { "self" : " https://api.server.com/teams/4676 " , "players" : " https://api.server.com/teams/4676/players " , "teamDetails" : " https://api.server.com/teamDetails/4676 " }  } 

翻译自: https://www.javacodegeeks.com/2019/06/defining-resource.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/341598.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

E1 PCM复用设备能当程控交换机用吗?

E1 PCM复用设备能当程控交换机用吗?这个问题在平时与用户沟通中发现,发现有不少用户认为E1 PCM复用设备可以当程控交换机使用,不仅能实现电话线远端延伸,还可以实现电话机交换功能。 其实这个观点是错误的,E1 PCM复用…

PCM设备能在公网使用吗?

正如标题所言,PCM设备能否在公网使用吗?首先我们可以了解一下公网的概念。“公网”也可以理解为“外网”,就是指公共的网络,用户可以使用这个网络相互访问对方。 其实,笔者认为“PCM能在公网使用”或“PCM不能在公网使…

日月神话_常见的Java神话

日月神话这些问题在任何面试中都可能太过高级而无法提出,因为它们可能会使候选人推迟。 不管怎样,他们可能会在您自己的时间练习。 误解1)System.exit(0)阻止最终被调用 为什么这样的代码 System.setSecurityManager…

PCM设备是做什么的? PCM作用是什么?

PCM可以提供用户从2M到155M速率的数字数据专线业务,也可以提供话音、图象传送、远程教学等其他业务。那么,PCM设备是做什么的? PCM作用是什么?接下来我们就跟随飞畅科技的小编一起来看看吧! PCM的含义: 1、PCM&…

Iceberg:浅析基于Snapshot的事务过程

MVCC事务(乐观锁) 我们知道,Iceberg基于Snapshot(快照机制)实现了乐观无锁地数据并发读写能力(MVCC,Multi Versions Concurrency Controll),默认提供了快照级别的事务隔…

关于“最终”的最终决定

在Java中,过度使用final感觉就像是SHOUTING。 在很多时候它已经过时且不合适。 Java和JavaScript 这主要是关于Java中的final关键字,但是我对JavaScript中与之对应的const看法略有变化。 我可以认为const是一件好事,而final是一件坏事&#…

PCM设备在电网系统中的应用介绍

近年来,随着我国社会经济的快速发展和电力通信行业的不断进步,通信网络技术和网络系统不断完善和改进的同时,网络系统运行过程中也出现了很多的问题,要对其进行优化,否则可能影响电力通信网络的正常运行电力通信网络系…

PCM设备在雷达通信系统应用

PCM设备系统是目前国内企业最常用的综合接入通信设备,PCM设备所具优势众多,因此它能成为目前企业最常用的通信传输接入设备之一,而且在类似雷达通信系统等重要行业中广泛应用着。 PCM设备利用标准的E1数据传输通道,采用PCM制式&a…

ejb的maven_针对新手的Java EE7和Maven项目-第3部分-定义ejb服务和jpa实体模块

ejb的maven从前面的部分恢复 第1部分 第2部分 我们在第三部分继续介绍,我们已经有一个父pom,并且已经为我们的war模块定义了pom。 在我们的原始设置中,我们定义了我们的应用程序将包含一个ejb jar形式的服务jar。 这是我们的Enterprise Jav…

PCM设备终端烧毁板卡是什么原因?

最近,有新客服反映他们之前在某家设备厂家买了PCM设备,老是会烧毁终端板。所以另外选择了我们飞畅科技。那么,PCM设备终端烧毁板卡是什么原因呢?接下来飞畅科技的小编就来为大家具体分析下PCM设备终端烧毁板卡的原因,一…

全部隐藏!

在较早的文章中 ,我写了以下几句话: 在面向对象的代码库中,该工具包应尽可能离散。 使用开发套件的次数越多,您的代码实际面向对象的次数就越少,或者您的抽象并不是最好的。 。 我认为有必要详细说明这一点&#xff0…

E1 PCM设备的主要特点介绍

E1 PCM设备是目前国内企业最常用的综合接入通信设备,它利用标准的E1数据传输通道,采用PCM制式,直接提供语音、数据、图像等多种用户接口。接下来飞畅科技的小编来为大家详细介绍下E1 PCM设备的主要特点,一起来看看吧! …

docker集群_使用Docker,Chef和Amazon OpsWorks进行集群范围的Java / Scala应用程序部署...

docker集群Docker非常适合在单个节点上运行隔离的容器。 但是,大多数软件系统都在多个节点上运行,因此,除了Docker之外,我们还需要某种方法来指定哪些容器应在哪些节点上运行。 我要解决的特定问题如下:我有两个Scala…

PCM信号是什么信号?

PCM信号是模拟信号呢,还是数字信号呢?当然是数字信号啦!PCM的完整定义是:将模拟信号的抽样量化值变换成代码称为脉冲编码调制(PCM设备)。 在光纤通信系统中,光纤中传输的是二进制光脉冲“0”码…

SDH光传输设备是什么?SDH设备特点介绍

SDH光传输设备是一种,将复接、线路传输以及交换功能融合为一体的、并且由统一网管系统操作的综合信息传送网络。SDH光传输设备,他可实现网络有效管理、能实时业务监控、能动态网络维护、不同厂商设备间的互通等多项功能。 SDH光传输设备能大大提高网络资…

通过MicroProfile上下文传播增强了CDI上下文和隔板

当将CDI与异步执行方法(例如ManagedExecutorService ,传统上不可能访问在原始线程中处于活动状态的所有CDI范围。 MicroProfile Context Propagation可以定义线程执行上下文并将其传递到完成阶段,尽管我们的代码是异步执行的,但它…

ProtoBuff3.0.0在Ubuntu上安装

ProtoBuff3.0.0在Ubuntu上安装 最近安装ns3,运行别人代码,编译build.py时出现需要update Proto的问题,本来想安装最新版本,因为需要翻墙,就参考大部分博客,安装3.0.0版本。(备注:直…

SDH与PDH的区别介绍

PDH准同步数字系列,1-4次群2048K、8488K、34368K、139264K,有美国/日本标准和欧洲标准,我国沿用的是欧洲标准;30/32路组成一个1次群;各个厂家产品大多互相不兼容。 SDH同步数字系列,网元都带PDH接口&#…

Ubuntu 系统入门

1 Ubuntu 常用文件夹处理命令 1、 cd … //返回上一层目录 cd / //返回根文件夹 2、ls 列出文件名称 3、unzip XXX 对zip 压缩包解压 4、sudo baobab 进行磁盘空间分析 5、 利用vim修改只读文件内容 ::w !sudo tee % (注意空格) 2 Ubuntu显示隐藏文件 …

什么是MSTP(多业务传输平台)?

Multi-Service Transfer Platform简称MSTP,他是基于SDH的多业务传送平台的,同时也实现TDM、ATM、以太网等业务的接入、处理和传送,提供统一网管的多业务节点。 当前通信时代随着不断增长的IP数据、话音、图像等多种业务传送需求扩展&#xf…