Web API开发中的RPC和REST

目录

RPC

REST

REST优点

REST缺点

Restful如何传递参数

三种方式的不同语义

建议

Web API如何返回状态码

问题:

400观点

200观点

建议

ASP.NET Core中REST落地指南

实现建议

实现

注意


Web API开发有两种风格:面向过程(RPC)、面向REST

RPC

业务驱动,自然
“控制器/操作方法“的形式把服务器端的代码当成方法去调用。把HTTP当成传输数据的通道,不关心HTTP谓词。通过QueryString、请求报文体给服务器传递数据。状态码。比如:/Persons/GetAll、/Persons/GetById?id=8、/Persons/Update、/Persons/DeleteById/8;

REST

要求开发人员对REST原则更了解、并且有更多的设计能力。
按照HTTP的语义来使用HTTP协议:
1、URL用于资源的定位:/user/888、/user/888/orders;
2、HTTP谓词:GET、POST(新增)、PUT(整体更新)、DELETE、PATCH(局部更新)等;
3、什么是“幂等”,举例?DELETE、PUT、GET是幂等的,POST不幂等;
4、GET的响应可以被缓存;
5、服务器端要通过状态码来反映资源获取的结果:404、403(没有权限)、201(新增成功)。

REST优点

  1. 通过URL对资源定位,语义更清晰;
  2. 通过HTTP谓词表示不同的操作,接口自描述;
  3. 可以对GET、PUT、DELETE请求进行重试;
  4. 可以用GET请求做缓存;
  5. 通过HTTP状态码反映服务器端的处理结果,统一错误处理机制。
  6. 网关等可以分析请求处理结果。

REST缺点

  1. 真实系统中的资源非常复杂,很难清晰地进行资源的划分,对技术人员的业务和技术水平要求高。
  2. 不是所有的操作都能简单地对应到确定的HTTP谓词中。
  3. 系统的进化可能会改变幂等性。
  4. 通过URL进行资源定位不符合中文用户的习惯。
  5. HTTP状态码个数有限。
  6. 有些环节会篡改非200响应码的响应报文。
  7. 有的客户端不支持PUT、DELETE请求。

Restful如何传递参数

  1. URL:适合定位;长度限制。
  2. QueryString:灵活;长度限制。
  3. 请求报文体:灵活;长度不限制;不支持GET、Delete。

三种方式的不同语义

URL:资源定位。
QueryString:URL之外的额外数据。
请求报文体:供PUT、POST提供数据。

建议

  1. 对于保存、更新类的请求POST、PUT请求,把全部参数都放到请求报文体中;
  2. 对于DELETE请求,要传递的参数就是一个资源的id,因此把参数放到QueryString中即可;
  3. 对于GET请求,一般参数的内容都不会太长,因此统一通过QueryString传递参数就可以。对于极少数参数内容超过URL限制的请求,由于GET、PUT请求都是幂等的,因此我们把请求改成通过PUT请求,然后通过报文体来传递参数。

Web API如何返回状态码

REST通过HTTP状态码返回服务器端的处理结果。

问题:

1)HTTP状态码数量有限;
2)HTTP的状态码并不适合用来表示业务层面的错误码,它是一个用来表示技术层面信息的状态码。新增用户的操作中,如果服务器端要求Json格式,客户端提交XML,服务器返回400是没问题的。但是如果用户名格式错误或者用户名重复,存在200派和400派。

400观点

网关等可以监控HTTP状态码,如果接口频繁出现4xx状态码,那么就说明客户端的代码不完善。

很多的系统都是按照HTTP状态码的不同含义进行设计的。如果失败了服务器端返回的状态码还是200的话,这会违背软件设计的初衷。

200观点

网络的问题归网络、业务的问题归业务。业务错误不应该和技术错误混在一起。把系统日志和业务日志区分开。

建议

  1. 对于数据库服务器连接失败、请求报文格式、服务器端异常等非业务错误,服务器端应该返回4xx、5xx等状态码。
  2. 对于业务层面的错误,比如用户不存在,我们要使用4xx等HTTP状态码返回。同样在响应报文体中给出详细的错误信息,比如{“code”:3,”message”:”用户不存在”}。
  3. 不仅要返回4xx的HTTP状态码,而且不要忘了通过响应报文体给出详细的错误信息。对于用户不存在,不仅要404,而且把响应报文体设置为{“code”:3,”message”:”用户名已存在”},这样能区分出来是哪里的问题。

ASP.NET Core中REST落地指南

实现建议

  1. URL有能力用REST,否则使用RPC风格:Users/AddNew、Users/GetAll、Users/DeleteById。
  2. 对于可以缓存的操作,使用GET请求;对于幂等的更新操作,使用PUT请求;对于幂等的删除操作,使用DELETE请求;对于其他操作,统一使用POST请求。
  3. 参数:保存、更新类的请求使用POST、PUT请求,把全部参数都放到请求报文体中;对于GET和DELETE请求,把参数放到QueryString中。推荐尽量使用URL做资源定位。
  4. 对于业务错误,服务器端返回合适的4XX状态码,不知道该选择哪个状态码就用400;同时,在报文体中通过code参数提供业务错误码以及错误消息。
  5. 如果请求的处理执行成功,服务器端返回值为200的Http状态码,如果有需要返回给客户端的数据,则服务器端把这些数据放到响应报文体

实现

  1. 控制器上 [Route("[controller]/[action]")]
  2. 强制要求控制器中不同的操作用不同的方法名
  3. 把[HttpGet]、[HttpPost]、[HttpDelete]、[HttpPut]等添加到对应的操作方法上。

注意

如果控制器中存在一个没有添加[HttpGet]、[HttpPost]等的public方法,Swagger就会报错,可以用[ApiExplorerSettings(IgnoreApi = true)]

你可能感兴趣的:(.NET,.netcore,网络协议)