定义资源

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Sentinel 是阿里巴巴开源的一个轻量级流量控制框架,它允许开发人员通过定义规则,对应用程序的资源进行保护和限制,防止系统出现因流量过大而导致的雪崩效应。下面是使用 Sentinel 定义资源的方式: 1. 引入 Sentinel 的依赖包,比如 `sentinel-core`。 2. 在应用程序中定义需要进行限流、熔断、降级等操作的资源,比如一个方法或者一个接口。 3. 使用 `@SentinelResource` 注解标注该资源,并设置相关的规则,比如流控规则、降级规则等。 4. 配置 Sentinel 控制台,将应用程序注册到 Sentinel 控制台中,并配置相应的规则。 5. 运行应用程序,Sentinel 就会按照规则对资源进行保护和限制,保证系统的稳定性和可靠性。 示例代码如下: ```java //定义需要保护的资源 @SentinelResource("sayHello") public String sayHello(String name) { return "Hello, " + name; } //设置流控规则 FlowRule flowRule = new FlowRule(); flowRule.setResource("sayHello"); flowRule.setGrade(RuleConstant.FLOW_GRADE_QPS); flowRule.setCount(10); //将规则加载到 Sentinel 中 FlowRuleManager.loadRules(Collections.singletonList(flowRule)); ``` 上述代码中,我们定义了一个名为 `sayHello` 的资源,并使用 `@SentinelResource` 注解对其进行标注。同时,我们设置了一个流控规则,表示对 `sayHello` 接口进行 QPS 限流,最大请求数为 10。最后,我们将规则加载到 Sentinel 中,启用流控功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值