当前位置: 首页 > news >正文

RESTful API设计:从原则到Gin实现

背景与优势: 现代 Web 应用通常由前端和后端两部分组成,各种终端设备(手机、平板、桌面等)层出不穷,需要统一的通信机制。REST(Representational State Transfer)作为一种基于 HTTP 协议的架构风格,提供了一套成熟的 API 设计理论。它具有以下优势:

基于标准 HTTP 方法,设计简单直观

每个请求独立无状态,系统具有良好的可扩展性和可靠性

充分利用 HTTP 缓存提高性能,降低服务器压力

统一的资源接口实现跨语言、跨平台互操作性

核心设计原则

一个好的 RESTful API 应该考虑以下问题:

  1. 协议:建议使用更为安全的 https 协议,保证数据传输的加密与安全。
  2. 域名:应部署在专属域名下,例如 https://api.my.com,与主域名 https://my.com 隔离,如果项目非常简单,也可以放在主域名的子路径下 https://my.com/api/)。
  3. 版本控制:应该将版本号放入url中,例如 https://api.my.com/v1/newlist,方便后续迭代和兼容。
  4. 资源路径:在 RESTful API 架构中, 每个网址代表一种资源 , 所以网址中建议不能有动词, 只能有名词, 而且所用的名词往往与数据库的表格名对应。
  5. 数据请求
    • 常用
      • GET(SELECT) : 从服务器取出资源(一项或多项)
      • POST(CREATE) : 在服务器新建一个资源
      • PUT(UPDATE) : 在服务器更新资源(客户端提供改变后的完整资源)
      • DELETE(DELETE) : 从服务器删除资源
    • 不常用
      • HEAD: 获取资源的元数据
      • OPTIONS: 获取信息, 关于资源的哪些属性是客户端可以改变的
      • PATCH(UPDATE) : 在服务器更新资源(客户端提供改变的属性)
  6. HTTP 状态码:遵循标准的 HTTP 状态码规范返回结果。如成功获取或创建资源通常返回 200 OK

以上设计原则有助于构建清晰、一致且易用的接口,使开发者不查看文档也能根据 URL 和方法猜测出接口功能。例如,使用 /collection 返回资源列表、/collection/{id} 返回单个资源、POST /collection 创建资源等模式。

RESTful 架构的局限性与挑战

尽管 RESTful 架构风格具有诸多优点,但在实际应用中也存在一些局限和挑战。

  • RESTful 风格容易出现“过度获取”或“获取不足”的问题:客户端往往需要从资源端点获取超出所需的数据,导致带宽浪费;或者需要发起多次请求才能获取关联数据,增加延迟。

  • RESTful 通常采用同步请求方式,在高并发或实时性要求高的场景下可能造成性能瓶颈和阻塞。

  • 数据模型的变更会影响客户端,客户端与服务器的紧耦合程度较高,需要协同更新才能兼容新接口。

  • RESTful 严格依赖 HTTP 协议,对于某些极限性能或低延迟应用可能并非最佳选择。

  • RESTful 无状态性也意味着诸如认证和会话管理等需要额外机制(如 Token 传递),增加了实现复杂度。

  • RESTful 规范并未严格规定所有细节,团队需要约定 API 细节,良好的文档和统一风格是减少歧义的关键。

总的来说,随着应用需求的发展,GraphQL、gRPC、异步事件驱动等新兴技术出现,为复杂数据查询和实时通信提供了替代方案。RESTful 在大多数 CRUD 场景下仍然简单高效,但对于现代多设备、大数据量、低延迟的复杂应用,需要结合具体场景权衡选型。

Gin框架设计一组 RESTful API 风格接口

package mainimport ("github.com/gin-gonic/gin""net/http"
)type User struct {ID       string `json:"id"`Username string `json:"username"`Email    string `json:"email"`
}var users = []User{{ID: "1", Username: "user1", Email: "user1@example.com"},{ID: "2", Username: "user2", Email: "user2@example.com"},
}func main() {r := gin.Default()userGroup := r.Group("/users"){userGroup.GET("", getUsers)          // 获取用户列表userGroup.POST("", createUser)       // 创建新用户userGroup.GET("/:id", getUser)       // 获取特定用户userGroup.PUT("/:id", updateUser)    // 更新用户(全量)userGroup.PATCH("/:id", patchUser)   // 更新用户(部分)userGroup.DELETE("/:id", deleteUser) // 删除用户}r.Run(":80")
}// 获取用户列表
func getUsers(c *gin.Context) {c.JSON(http.StatusOK, users)
}// 创建用户
func createUser(c *gin.Context) {var newUser Userif err := c.ShouldBindJSON(&newUser); err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})return}users = append(users, newUser)c.JSON(http.StatusCreated, newUser)
}// 获取特定用户
func getUser(c *gin.Context) {id := c.Param("id")for _, user := range users {if user.ID == id {c.JSON(http.StatusOK, user)return}}c.JSON(http.StatusNotFound, gin.H{"message": "User not found"})
}// 全量更新用户
func updateUser(c *gin.Context) {id := c.Param("id")var updatedUser Userif err := c.ShouldBindJSON(&updatedUser); err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})return}for i, user := range users {if user.ID == id {users[i] = updatedUserc.JSON(http.StatusOK, updatedUser)return}}c.JSON(http.StatusNotFound, gin.H{"message": "User not found"})
}// 部分更新用户
func patchUser(c *gin.Context) {id := c.Param("id")var updates map[string]interface{}if err := c.ShouldBindJSON(&updates); err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})return}for i, user := range users {if user.ID == id {// 实际应用中应该使用反射或结构体处理部分更新if username, ok := updates["username"]; ok {users[i].Username = username.(string)}if email, ok := updates["email"]; ok {users[i].Email = email.(string)}c.JSON(http.StatusOK, users[i])return}}c.JSON(http.StatusNotFound, gin.H{"message": "User not found"})
}// 删除用户
func deleteUser(c *gin.Context) {id := c.Param("id")for i, user := range users {if user.ID == id {users = append(users[:i], users[i+1:]...)c.JSON(http.StatusNoContent, nil)return}}c.JSON(http.StatusNotFound, gin.H{"message": "User not found"})
}

设计了这六个接口用来获取或修改资源,在后端开发中应该是直接操作数据库,这里我们就使用直接的数据结构替代了

userGroup.GET("", getUsers)          // 获取用户列表
userGroup.POST("", createUser)       // 创建新用户
userGroup.GET("/:id", getUser)       // 获取特定用户
userGroup.PUT("/:id", updateUser)    // 更新用户(全量)
userGroup.PATCH("/:id", patchUser)   // 更新用户(部分)
userGroup.DELETE("/:id", deleteUser) // 删除用户
http://www.xdnf.cn/news/576325.html

相关文章:

  • 【AI模型学习】ESM2
  • 部署rsync远程同步+inotify监控
  • 前端学习(6)—— WebAPI部分案例
  • 前端面经-WebGL/threeJS
  • 《Saliency Attack: Towards Imperceptible Black-box Adversarial Attack》论文分享(侵删)
  • Spring AI 1.0 快速入门
  • NVIDIA GPU 性能调优与诊断完全指南
  • ConcurrentHashMap导致的死锁事故
  • 环境搭建
  • 根据Spring官方文档,三分钟完成Springboot项目集成Spring AI
  • sqli-labs第十七关——POST注入点
  • Spring Boot整合Redis
  • RestTemplate 发送的字段第二个大写字母变成小写的问题探究
  • 9-码蹄集600题基础python篇
  • leetcode 螺旋矩阵 java
  • 5-码蹄集600题基础python篇
  • 如何设计智慧工地系统的数据库?
  • 系统程序变更管理:确保IT环境稳定性和安全性的关键
  • Entity-Relationship Model(实体-关系模型)
  • FlashAttention:传统自注意力( Self-Attention)优化加速实现
  • 用户刷题记录日历——签到表功能实现
  • 基于 Guns v5.1 框架的分页教程
  • SseEmitter是什么
  • 卷积神经网络基础(十)
  • chrono类 根据duration 类的周期类型得到对应的周期名称
  • 预警功能深度测评:如何用系统降低设备突发故障率?
  • JavaScript常用事件
  • 第P10周:Pytorch实现车牌识别
  • 如何解决测试覆盖率与迭代速度的冲突问题?
  • 手搓四人麻将程序