第九篇 API设计原则与最佳实践

文章目录

  • API设计原则与最佳实践
    • RESTful API设计原则
      • 1. 资源导向
        • a. 资源识别
        • b. HTTP动词的使用
      • 2. 状态转移
      • 3. 统一接口
      • 4. RESTful API 示例(axios调用)
      • 5. 非RESTful API 示例(axios调用)
    • HATEOAS理念与实践
      • 1. Hypermedia as the Engine of Application State (HATEOAS)
        • a. 原则阐述
        • b. 实践举例
    • API版本控制策略
      • 1. 版本管理方法
        • i. URL路径版本化
        • ii. 请求头版本化
        • iii. 查询参数版本化
    • 错误处理与反馈设计
      • 1. 错误处理机制
        • a. HTTP状态码
        • b. 错误响应格式

API设计原则与最佳实践

RESTful API设计原则

1. 资源导向

a. 资源识别

在REST架构风格中,API的设计围绕资源展开。每个URL代表一个可被唯一标识的资源,并且应当清晰、简洁和语义化。例如:

/api/users       # 用户集合资源
/api/users/42    # 特定ID为42的用户资源
b. HTTP动词的使用

通过HTTP方法来定义对资源的操作:

  • GET:用于获取资源信息。
  • POST:用于创建新资源。
  • PUT:用于替换整个资源。
  • PATCH:用于更新资源的部分内容。
  • DELETE:用于删除资源。

2. 状态转移

API应遵循无状态通信原则,每次请求都应包含所有必要的信息以完成操作,服务器不保存客户端会话状态。

3. 统一接口

确保API的一致性,不同资源类型的处理方式应当遵从相同的模式。

为了更直观地体现RESTful API的优势,我们将通过使用axios库(一个流行的JavaScript HTTP客户端)来调用RESTful API和非RESTful API的示例进行对比。

4. RESTful API 示例(axios调用)

// 引入axios库
import axios from 'axios';// 创建RESTful API实例
const api = axios.create({baseURL: 'https://api.example.com/v1',
});// 调用RESTful API资源
async function createNewOrder(orderData) {try {// POST请求创建新订单const response = await api.post('/orders', orderData);return response.data;} catch (error) {console.error('Error creating order:', error.response.data);}
}async function updateOrderStatus(orderId, newStatus) {try {// PUT请求更新订单状态const url = `/orders/${orderId}`;const data = { status: newStatus };const response = await api.put(url, data);return response.data;} catch (error) {console.error('Error updating order status:', error.response.data);}
}async function deleteOrder(orderId) {try {// DELETE请求删除订单const url = `/orders/${orderId}`;await api.delete(url);} catch (error) {console.error('Error deleting order:', error.response.data);}
}// 使用示例
const newOrder = { /* 订单数据 */ };
createNewOrder(newOrder).then(createdOrder => console.log('Created order:', createdOrder));const orderIdToUpdate = '123';
const updatedStatus = 'shipped';
updateOrderStatus(orderIdToUpdate, updatedStatus).then(updatedOrder => console.log('Updated order:', updatedOrder));const orderIdToDelete = '456';
deleteOrder(orderIdToDelete).then(() => console.log('Deleted order with ID:', orderIdToDelete));

5. 非RESTful API 示例(axios调用)

// 同样引入axios库
import axios from 'axios';// 非RESTful API的调用方式
const nonRestApi = axios.create({baseURL: 'https://legacy.example.com/api',
});async function executeNonRestAction(action, data) {try {// 动作类型作为URL的一部分const url = `/${action}`;const response = await nonRestApi.post(url, data);return response.data;} catch (error) {console.error(`Error executing action ${action}:`, error.response.data);}
}async function createLegacyOrder(orderData) {try {const response = await executeNonRestAction('CreateNewOrder', orderData);return response;} catch (error) {console.error('Error creating legacy order:', error.response.data);}
}async function updateLegacyOrderStatus(orderId, newStatus) {try {const requestData = { orderId, status: newStatus };const response = await executeNonRestAction('UpdateOrderStatus', requestData);return response;} catch (error) {console.error('Error updating legacy order status:', error.response.data);}
}// 使用示例
const legacyOrder = { /* 订单数据 */ };
createLegacyOrder(legacyOrder).then(createdOrder => console.log('Created legacy order:', createdOrder));const legacyOrderId = '789';
const legacyUpdatedStatus = 'processing';
updateLegacyOrderStatus(legacyOrderId, legacyUpdatedStatus).then(updatedOrder => console.log('Updated legacy order:', updatedOrder));

优势对比:

  • 清晰性与一致性:RESTful API中每个HTTP方法都对应着特定的操作意图,使得API的设计更加直观且易于理解。例如,在RESTful API中,POST /orders明确表示创建新订单,而PUT /orders/:id则用于更新订单,结构一致,无需额外记忆特殊动作名。

  • 可扩展性与维护性:随着系统的发展,RESTful API可以轻松添加新的资源或资源操作,因为它们遵循统一的模式。而非RESTful API在新增功能时可能需要不断调整URL路径和动作命名规则。

  • 工具支持与缓存机制:许多开发工具和网络基础设施针对RESTful API设计提供了更好的支持,如自动补全、接口文档生成器等。此外,RESTful API更容易利用HTTP协议的缓存特性,如响应头中的Cache-Control、ETag等字段,提升性能。

  • 版本控制与兼容性:RESTful API可以通过URI路径或Accept头等方式实现版本控制,从而方便地处理升级过程中的兼容问题。非RESTful API在没有明确版本管理的情况下可能会带来更复杂的升级挑战。

通过上述对比,可以看出RESTful API在设计原则、开发者体验、长期维护以及与现有Web标准的契合度等方面具有显著优势。

HATEOAS理念与实践

1. Hypermedia as the Engine of Application State (HATEOAS)

a. 原则阐述

HATEOAS是REST的核心特性之一,它意味着API响应中应该包含足够的链接信息,使得客户端可以通过这些链接发现接下来可以执行的动作,而无需预知具体URL。

例如,在一个电商应用中,当客户端请求一个订单资源时,服务器返回的JSON响应不仅包含订单详情,还会包括链接到相关操作的URL,如:

{"id": "12345","customer": "John Doe","status": "pending","total": "$100.00","_links": {"self": { "href": "/orders/12345" },"cancel": { "href": "/orders/12345/cancel", "method": "POST" },"pay": { "href": "/orders/12345/pay", "method": "POST" },"items": { "href": "/orders/12345/items" }}
}

在这个例子中,_links属性定义了几个链接:

  • self:指向当前订单资源自身的URL。
  • cancelpay:分别指示了取消订单和支付订单的操作,通过POST方法触发这些动作,并给出了对应的动作URL。
  • items:指向与当前订单相关的商品列表资源。

这样的设计允许客户端根据接收到的数据结构动态发现并导航到不同的资源和状态,无需硬编码URL或者了解整个API的结构。这使得API更加灵活、可扩展且易于维护,同时增强了客户端的适应性,因为它们可以根据服务器提供的上下文信息来决定下一步的操作。

b. 实践举例
{"user": {"id": 42,"name": "Alice","_links": {"self": { "href": "/api/users/42" },"edit": { "href": "/api/users/42/edit" },"orders": { "href": "/api/users/42/orders" }}}
}

在这个示例中,API返回的数据包含了指向当前用户资源、编辑资源以及该用户订单集合的链接,允许客户端通过解析这些链接来进行导航和交互。

在遵循HATEOAS原则的RESTful API设计中,_links或类似结构中的键(keys)并不是随意生成的,而是根据资源间关系和操作意图来命名的。例如,“self”、“next”、“prev”、“create”、"update"等都是常见的链接关系名称,它们具有一定的语义意义,客户端可以根据这些名称理解链接的目的。

对于前端来说,确实需要处理这一层额外的信息,但这样做的好处在于:

  1. 动态发现:客户端无需预先知道所有可能的接口地址或者API的变化,只需要解析返回的链接信息就能找到下一步操作的入口。

  2. 可扩展性:当API进行版本升级或添加新的功能时,只要保持链接关系的约定不变,前端应用可以继续正常工作,无需硬编码新的URL。

  3. 松耦合:前后端之间的耦合度降低,后端服务架构可以独立演化而不影响前端用户体验。

至于是否需要存储这个信息以便以后使用,这取决于具体的业务需求。有些情况下,客户端可能需要临时缓存部分链接以实现页面间的跳转或者异步操作;而在其他场景下,客户端可能会立即利用这些链接执行后续请求,并不需要持久化存储。通常,在构建响应式前端应用时,会根据当前状态和用户交互实时处理接收到的链接信息,而不是静态地存储全部链接数据。

API版本控制策略

1. 版本管理方法

i. URL路径版本化

将版本号嵌入到URL路径中,如:

/v1/users
/v2/users
ii. 请求头版本化

通过HTTP请求头传递版本信息,例如:

Accept: application/vnd.example-com.user+json;version=1.0
iii. 查询参数版本化

也可以选择将版本号作为查询参数传递:

/api/users?version=1

错误处理与反馈设计

1. 错误处理机制

a. HTTP状态码

正确使用HTTP状态码传达请求结果的状态,如4xx系列表示客户端错误,5xx系列表示服务器端错误。

b. 错误响应格式

提供统一且详细的错误消息体,通常是一个JSON对象,包括错误代码、描述信息、可能的话还包含解决问题的建议或指向帮助文档的链接。

{"error": "BadRequest","message": "Missing required parameter 'username'","status_code": 400,"more_info": "https://docs.example.com/errors/400"
}

通过遵循以上设计原则与最佳实践,开发者能够构建出易于理解和使用的RESTful API,并确保其在版本迭代和异常处理时具备良好的扩展性和可靠性。

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

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

相关文章

使用Flash_Download_Tool下载PlatformIO生成的bin程序到ESP32

使用Flash_Download_Tool下载PlatformIO生成的bin程序到ESP32 来源 当我们没有PlatformIO环境时,还要下载PlatformIO生成的程序时,可以使用Flash_Download_Tool工具下载。 说明 使用PlatformIO时,用cmd终端命令下载程序pio run -v -t upl…

2023 IoTDB Summit:天谋科技高级开发工程师张金瑞《筑其形:如何轻松搞定 IoTDB 数据建模》...

12 月 3 日,2023 IoTDB 用户大会在北京成功举行,收获强烈反响。本次峰会汇集了超 20 位大咖嘉宾带来工业互联网行业、技术、应用方向的精彩议题,多位学术泰斗、企业代表、开发者,深度分享了工业物联网时序数据库 IoTDB 的技术创新…

CSS中用border画三角形等图形

问题背景 最近在做一个官网项目的时候,UI小姐姐设计了一些特殊图形,例如三角形,虽然element-ui是带有相应模板的,但是设计的和模版的不一样,因此研究了一下。 三角形怎么画 三角形实际上是border的产物 我们正常使用…

SpringBoot 把PageHelper分页信息返回给前端

第1步&#xff1a;定义线程容器收纳HttpHeaders和HttpStatus import org.springframework.http.HttpHeaders; import org.springframework.http.HttpStatus;public class ResponseUtils {private static ThreadLocal<HttpHeaders> ThreadLocalHeaders new InheritableT…

Window平台安装MongoDB

在部署前需要在官网先确定系统对应的Mongo DB版本。 本机电脑为Window10&#xff0c;所以这里以MongoDB 6.0版本。 1 在官网下载安装包 2 安装MongoDB MongoDB Compass 是一个图形界面管理工具&#xff0c;如果勾选了安装会花费长一点时间&#xff0c;可以取消掉勾选&#xff…

构建基于RHEL8系列(CentOS8,AlmaLinux8,RockyLinux8等)的Nginx1.24.0的RPM包

本文适用&#xff1a;rhel8系列&#xff0c;或同类系统(CentOS8,AlmaLinux8,RockyLinux8等) 文档形成时期&#xff1a;2022-2023年 因系统版本不同&#xff0c;构建部署应略有差异&#xff0c;但本文未做细分&#xff0c;对稍有经验者应不存在明显障碍。 因软件世界之复杂和个人…

Dockerfile: CMD与ENTRYPOINT区别

CMD和ENTRYPOINT的作用 CMD和ENTRYPOINT这两个命令&#xff0c;我接触到的是用在了Dockerfile中用于构建容器。 CMD&#xff1a;The main purpose of a CMD is to provide defaults for an executing container. CMD的主要用途是为正在执行的容器提供默认值。也就是指定这个容…

web前端算法简介之栈

栈 栈的基本操作包括&#xff1a; 初始化栈&#xff08;InitStack&#xff09;&#xff1a;判断栈是否为空&#xff08;IsStackEmpty&#xff09;&#xff1a;入栈&#xff08;Push&#xff09;&#xff1a;出栈&#xff08;Pop&#xff09;&#xff1a;获取栈顶元素&#xff0…

如何将ArcGIS工程文件迁移到ArcGIS Pro内

当你刚接触ArcGIS Pro的时候&#xff0c;尝试新建一个工程文件会发现工程文件的后缀已经改变&#xff0c;那么以前在ArcGIS内辛苦制作的工程文件是否就不能在ArcGIS Pro内使用了&#xff0c;答案是否定的&#xff0c;对此Esri也给出了解决方案&#xff0c;这里为大家介绍一下迁…

MergeTwoSortedLists 【合并有序链表】

有种归并排序的感觉 链表好久不用有些生疏了&#xff0c;思想思路是对的&#xff0c;但是代码写出来有问题&#xff0c;。 写完说点感受&#xff1a; 当时在学校学习链表的时候&#xff0c;就了解到链表分为“有头节点”和“无头节点”的链表&#xff0c;所以这里好像就不练&am…

Docker数据持久化

数据持久化 在容器层的 UnionFS&#xff08;联合文件系统&#xff09;中对文件/目录的任何修改&#xff0c;无论是手工修改还是 容器在运行过程中的修改&#xff0c;在该容器丢失或被删除后这些修改将全部丢失。即这些修改是无 法保存下来的。若要保存下来这些修改&#xff0c;…

如何一键添加引号和英文逗号,然后可以放入SQL中使用 → WHERE USER_NAME IN (‘张三‘,‘李四‘,‘王五‘)

如何一键添加引号和英文逗号&#xff0c;然后可以放入SQL中使用 → WHERE USER_NAME IN&#xff08;张三,李四,王五&#xff09; 一、背景二、解决方法三、一键添加引号和英文逗号的教程 一、背景 在日常开发中&#xff0c;当处理VARCHAR或VARCHAR2类型的字段时&#xff0c;很…

2024,AI Agent的密集爆发之年

最近这几天&#xff0c;相信已经有很多朋友看到了关于GPT Store、Vision Pro、Rabbit R1、AI pin、英伟达ACE&#xff08;Avatar Cloud Engine&#xff09;、钉钉个人助理、荣耀MagicOS 8.0等各类和AI技术深度结合的AI Agent或者承载AI Agent的平台。有些是和个人应用相关&…

写一个简单的Java的Gui文本输入窗口,JFrame的简单使用

JFrame是指一个计算机语言-java的GUI程序的基本思路是以JFrame为基础,它是屏幕上window的对象,能够最大化、最小化、关闭。 Swing的三个基本构造块:标签、按钮和文本字段;但是需要个地方安放它们,并希望用户知道如何处理它们。JFrame 类就是解决这个问题的——它是一个容器…

【webpack学习】webpack插件之“html-webpack-plugin”

1. 介绍 html-webpack-plugin 是一个Webpack插件&#xff0c;用于简化在构建过程中创建HTML文件的过程。它的作用是根据你的Webpack构建配置&#xff0c;在构建时自动生成一个或多个HTML文件&#xff0c;并将打包生成的JavaScript和CSS文件自动引入到这些HTML文件中。 2. 功能…

Redis在项目开发中的应用

Spring Boot集成Redis构建博客应用 在这个示例中&#xff0c;我们将展示如何使用Spring Boot和Redis构建一个简单的博客应用&#xff0c;包括文章发布、点赞和评论功能。 1. 添加依赖 首先&#xff0c;我们需要在pom.xml文件中添加Spring Boot和Redis的依赖项。 <!-- Sp…

docsify阿里云上部署

使用Markdown格式安装和部署Nginx 本文将介绍如何使用Markdown格式安装和部署Nginx。 步骤 安装Nginx&#xff1a; 打开终端&#xff0c;并根据您的操作系统执行以下命令来安装Nginx&#xff1a; 对于Ubuntu或Debian系统&#xff1a; sudo apt-get update sudo apt-get insta…

损失函数篇 | RT-DETR 引入 Inner-IoU 考虑边框形状与尺度的度量

作者导读:Inter-IoU:基于辅助边框的IoU损失 论文地址:https://arxiv.org/abs/2311.02877 作者视频解读:https://www.bilibili.com 开源代码地址:https://github.com/malagoutou/Inner-IoU 随着探测器的快速发展,边界框回归(BBR)损失函数不断更新和优化。然而,现有基…

C++进阶(三)多态

&#x1f4d8;北尘_&#xff1a;个人主页 &#x1f30e;个人专栏:《Linux操作系统》《经典算法试题 》《C》 《数据结构与算法》 ☀️走在路上&#xff0c;不忘来时的初心 文章目录 一、多态的概念1、概念 二、多态的定义及实现1、多态的构成条件2、虚函数3、虚函数的重写4、C…

酒茶元宇宙:探索未来生活的无限可能

随着科技的飞速发展&#xff0c;人类对于未来生活的想象也在不断地拓展。在这个过程中&#xff0c;茶酒文化作为一种源远流长的传统文化&#xff0c;也在逐渐融入现代科技&#xff0c;为人们带来全新的生活体验。茶酒元宇宙&#xff0c;就是这样一个将茶酒文化与科技相结合的全…