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

【9.2】Golang后端开发系列--Gin路由定义与实战使用

文章目录

      • 一、Gin 框架路由的基本定义方式
        • 1. 简单路由创建
        • 2. 路由参数
        • 3. 查询参数
      • 二、商业大项目中的路由定义和服务调用
        • 1. 路由模块化
        • 2. 路由组和中间件
        • 3. 中间件的使用
        • 4. 服务层调用
        • 5. 错误处理
        • 6. 版本控制
        • 7. 路由注册

一、Gin 框架路由的基本定义方式

1. 简单路由创建

使用 gin.Default() 创建一个带有默认中间件的路由引擎,然后通过 GET, POST, PUT, DELETE 等方法来定义不同 HTTP 方法的路由。

package mainimport ("github.com/gin-gonic/gin""net/http"
)func main() {// 创建带有默认中间件的路由引擎r := gin.Default()// 定义 GET 请求的路由r.GET("/hello", func(c *gin.Context) {c.String(http.StatusOK, "Hello, World!")})// 启动服务r.Run(":8080")
}
2. 路由参数

使用 : 来定义路由参数,通过 c.Param() 方法从请求中提取参数。

r.GET("/users/:userID", func(c *gin.Context) {userID := c.Param("userID")c.String(http.StatusOK, "User ID: %s", userID)
})
3. 查询参数

使用 c.Query() 方法来获取查询参数。

r.GET("/search", func(c *gin.Context) {query := c.Query("q")c.String(http.StatusOK, "Search query: %s", query)
})

二、商业大项目中的路由定义和服务调用

1. 路由模块化

为了使代码结构更清晰和易于维护,将路由定义分散到不同的文件和包中。

routes/user.go

package routesimport ("github.com/gin-gonic/gin""yourproject/services"
)func SetupUserRoutes(r *gin.RouterGroup) {r.GET("/", services.GetAllUsers)r.GET("/:id", services.GetUserByID)r.POST("/", services.CreateUser)r.PUT("/:id", services.UpdateUser)r.DELETE("/:id", services.DeleteUser)
}

services/user.go

package servicesimport ("github.com/gin-gonic/gin""net/http"
)func GetAllUsers(c *gin.Context) {// 调用数据访问层或其他服务层的函数来获取用户列表users, err := userService.GetAll()if err!= nil {c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to fetch users"})return}c.JSON(http.StatusOK, users)
}func GetUserByID(c *gin.Context) {id := c.Param("id")user, err := userService.GetByID(id)if err!= nil {c.JSON(http.StatusNotFound, gin.H{"error": "User not found"})return}c.JSON(http.StatusOK, user)
}func CreateUser(c *gin.Context) {var newUser Userif err := c.BindJSON(&newUser); err!= nil {c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid input"})return}if err := userService.Create(newUser); err!= nil {c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to create user"})return}c.JSON(http.StatusCreated, gin.H{"message": "User created successfully"})
}func UpdateUser(c *gin.Context) {id := c.Param("id")var updatedUser Userif err := c.BindJSON(&updatedUser); err!= nil {c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid input"})return}if err := userService.Update(id, updatedUser); err!= nil {c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to update user"})return}c.JSON(http.StatusOK, gin.H{"message": "User updated successfully"})
}func DeleteUser(c *gin.Context) {id := c.Param("id")if err := userService.Delete(id); err!= nil {c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to delete user"})return}c.JSON(http.StatusOK, gin.H{"message": "User deleted successfully"})
}
2. 路由组和中间件

使用路由组来组织相关的路由,并且可以为路由组添加中间件。

routes/router.go

package routesimport ("github.com/gin-gonic/gin""yourproject/middleware""yourproject/services"
)func SetupRoutes() *gin.Engine {r := gin.Default()// 公共中间件,例如日志、认证等r.Use(middleware.Logger(), middleware.Authenticate())// 用户路由组users := r.Group("/users")SetupUserRoutes(users)// 产品路由组products := r.Group("/products")SetupProductRoutes(products)return r
}
3. 中间件的使用

中间件可以在请求处理的不同阶段执行逻辑,如日志记录、认证、授权、错误处理等。

middleware/logger.go

package middlewareimport ("github.com/gin-gonic/gin""log""time"
)func Logger() gin.HandlerFunc {return func(c *gin.Context) {start := time.Now()c.Next()duration := time.Since(start)log.Printf("%s %s %s %v", c.Request.Method, c.Request.URL.Path, c.Writer.Status(), duration)}
}

middleware/authenticate.go

package middlewareimport ("github.com/gin-gonic/gin""net/http"
)func Authenticate() gin.HandlerFunc {return func(c *gin.Context) {// 检查用户是否认证,例如通过 token 验证if!isAuthenticated(c) {c.AbortWithStatusJSON(http.StatusUnauthorized, gin.H{"error": "Unauthorized"})return}c.Next()}
}func isAuthenticated(c *gin.Context) bool {// 实际的认证逻辑,例如检查请求头中的 tokenauthHeader := c.GetHeader("Authorization")if authHeader == "" {return false}// 更复杂的验证逻辑,如验证 token 的有效性return true
}
4. 服务层调用

将业务逻辑封装在服务层,路由处理函数负责调用相应的服务。

services/user_service.go

package servicesimport ("errors"
)type UserService struct{}func (s *UserService) GetAll() ([]User, error) {// 模拟从数据库或其他数据源获取用户列表// 这里应该调用实际的数据访问层函数users := []User{{ID: "1", Name: "Alice"},{ID: "2", Name: "Bob"},}return users, nil
}func (s *UserService) GetByID(id string) (*User, error) {// 模拟根据 ID 获取用户// 这里应该调用实际的数据访问层函数if id == "1" {return &User{ID: "1", Name: "Alice"}, nil}return nil, errors.New("User not found")
}func (s *UserService) Create(user User) error {// 模拟创建用户// 这里应该调用实际的数据访问层函数return nil
}func (s *UserService) Update(id string, user User) error {// 模拟更新用户// 这里应该调用实际的数据访问层函数return nil
}func (s *UserService) Delete(id string) error {// 模拟删除用户// 这里应该调用实际的数据访问层函数return nil
}
5. 错误处理

在路由处理函数和服务层中都要进行错误处理,确保将错误信息正确返回给客户端。

func GetUserByID(c *gin.Context) {id := c.Param("id")user, err := userService.GetByID(id)if err!= nil {c.JSON(http.StatusNotFound, gin.H{"error": err.Error()})return}c.JSON(http.StatusOK, user)
}
6. 版本控制

对于 API 的不同版本,可以使用路由组进行管理。

func SetupRoutes() *gin.Engine {r := gin.Default()v1 := r.Group("/v1")SetupV1Routes(v1)v2 := r.Group("/v2")SetupV2Routes(v2)return r
}
7. 路由注册

main 函数中注册路由并启动服务。

package mainimport ("yourproject/routes"
)func main() {r := routes.SetupRoutes()r.Run(":8080")
}

通过上述方式,在商业大项目中使用 Gin 框架时,可以将路由定义、中间件、服务调用和错误处理等部分分离,实现清晰的分层架构,提高代码的可维护性和可扩展性。这样做有助于团队协作开发,并且方便对不同功能模块进行单独测试和优化。同时,合理使用路由组和中间件可以确保不同路由的逻辑隔离和功能复用。

http://www.lryc.cn/news/521221.html

相关文章:

  • 【微信小程序】let和const-综合实训
  • 图匹配算法(涵盖近似图匹配)
  • java线程——Thread
  • MySQL8.0新特性
  • Oracle EBS GL定期盘存WIP日记账无法过账数据修复
  • 【绝对无坑】Mongodb获取集合的字段以及数据类型信息
  • 【Git版本控制器--1】Git的基本操作--本地仓库
  • C++并发编程之无锁数据结构及其优缺点
  • Ubuntu上,ffmpeg如何使用cuda硬件解码、编码、转码加速
  • rclone,云存储备份和迁移的瑞士军刀,千字常文解析,附下载链接和安装操作步骤...
  • Ubuntu | 系统软件安装系列指导说明
  • 队列(算法十三)
  • vLLM私有化部署大语言模型LLM
  • OpenAI Whisper:语音识别技术的革新者—深入架构与参数
  • 基于当前最前沿的前端(Vue3 + Vite + Antdv)和后台(Spring boot)实现的低代码开发平台
  • 【Rust】错误处理机制
  • Logback日志技术
  • 9分布式微服务架构
  • Leecode刷题C语言之统计重新排列后包含另一个字符串的子字符串数目②
  • HTML和CSS相关的问题,为什么页面加载速度慢?
  • LiveGBS流媒体平台GB/T28181常见问题-没有收到视频流播放时候提示none rtp data receive未收到摄像头推流如何处理?
  • Flask表单处理与验证
  • 正泰电工携手图扑:变电站数字孪生巡检平台
  • 瑞芯微 RK 系列 RK3588 使用 ffmpeg-rockchip 实现 MPP 视频硬件编解码-代码版
  • uniapp 预加载分包,减少loading
  • c#删除文件和目录到回收站
  • GESP2024年12月认证C++六级( 第三部分编程题(1)树上游走)
  • Redis数据结构服务器
  • 【向量数据库 Milvus】centos8源码安装和部署 Milvus 2.5.3
  • MySQL数据库(SQL分类)