【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 框架时,可以将路由定义、中间件、服务调用和错误处理等部分分离,实现清晰的分层架构,提高代码的可维护性和可扩展性。这样做有助于团队协作开发,并且方便对不同功能模块进行单独测试和优化。同时,合理使用路由组和中间件可以确保不同路由的逻辑隔离和功能复用。
相关文章:
【9.2】Golang后端开发系列--Gin路由定义与实战使用
文章目录 一、Gin 框架路由的基本定义方式1. 简单路由创建2. 路由参数3. 查询参数 二、商业大项目中的路由定义和服务调用1. 路由模块化2. 路由组和中间件3. 中间件的使用4. 服务层调用5. 错误处理6. 版本控制7. 路由注册 一、Gin 框架路由的基本定义方式 1. 简单路由创建 使…...
【微信小程序】let和const-综合实训
let 和 const 都是用于声明变量的关键字,它们与传统的 var 关键字相比,有很多不同之处。 let 声明块级作用域变量,可再赋值;const 声明块级作用域常量,不可再赋值。 以下是它们的详细介绍: 一、基本概念…...

图匹配算法(涵盖近似图匹配)
【图数据管理与挖掘-第四讲(子)图匹配算法(涵盖近似图匹配) 北京大学2021暑期-邹磊教授】https://www.bilibili.com/video/BV1zh411q7PW?vd_source7c2b5de7032bf3907543a7675013ce3a 图同构: 定义: 给定…...

java线程——Thread
java线程——Thread 基本步骤示例优劣总结 继承Thread类是Java中实现多线程的一种方式。使用时创建一个新的类,该类继承自java.lang.Thread,并重写其run()方法,在方法中定义线程执行的任务逻辑。 基本步骤 1、创建一个子类:定义一…...
MySQL8.0新特性
第十八章_MySQL8.0新特性 1.新特性概述 1. 数据库管理和存储 1.1 数据字典 特性: MySQL 8.0 使用统一的数据字典存储元数据(如表、列、索引等),并将其存储在 InnoDB 表中。 优点 : 提升性能:减少对文件系统的依赖。 提高一致…...

Oracle EBS GL定期盘存WIP日记账无法过账数据修复
系统环境 RDBMS : 12.1.0.2.0 Oracle Applications : 12.2.6 问题症状 用户反映来源为“定期盘存”和类别为“WIP”的日记账无法过账,标准日记账的界面上的过账按钮灰色不可用。但是,在超级用户职责下,该日记账又可以过账,细心检查发现该业务实体下有二个公司段值15100和…...
【绝对无坑】Mongodb获取集合的字段以及数据类型信息
Mongodb获取集合的字段以及数据类型信息 感觉很LOW的一个数据仓工具seatunel,竟然不能自动读取mongodb的表结构信息,需要手工创建。 然鹅,本人对mongodb也是新手,很多操作也不知所措,作为一个DBA,始终还是…...

【Git版本控制器--1】Git的基本操作--本地仓库
目录 初识git 本地仓库 认识工作区、暂存区、版本库 add操作与commit操作 master文件与commit id 修改文件 版本回退 撤销修改 删除文件 初识git Git 是一个分布式版本控制系统,主要用于跟踪文件的更改,特别是在软件开发中。 为什么要版本…...
C++并发编程之无锁数据结构及其优缺点
在C并发编程中,无锁数据结构(Lock-free Data Structures)是指那些在实现中不使用互斥锁(如std::mutex)来保证线程安全的数据结构。相反,它们利用原子操作和内存模型来确保多线程环境下的正确性和高效性。下…...

Ubuntu上,ffmpeg如何使用cuda硬件解码、编码、转码加速
本文使用 Ubuntu 环境。Ubuntu 直接使用 APT 安装的就支持 CUDA 加速。本文使用这样下载的版本进行演示,你自己编译或者其他源的版本可能会不同。 ffmpeg 的一些介绍,以及 macOS 版本的 ffmpeg 硬件加速请见《macOS上如何安装(不需要编译安装…...

rclone,云存储备份和迁移的瑞士军刀,千字常文解析,附下载链接和安装操作步骤...
一、什么是rclone? rclone是一个命令行程序,全称:rsync for cloud storage。是用于将文件和目录同步到云存储提供商的工具。因其支持多种云存储服务的备份,如Google Drive、Amazon S3、Dropbox、Backblaze B2、One Drive、Swift、…...
Ubuntu | 系统软件安装系列指导说明
文章目录 Ubuntu 系统软件安装系列指导说明工具系列1. Docker 与 Docker-Compose部署与安装 环境系列1. Golang部署与安装 数据库系列1. PostgreSQL17.2源码部署与安装 Ubuntu 系统软件安装系列指导说明 工具系列 1. Docker 与 Docker-Compose部署与安装 链接 环境系列 1…...

队列(算法十三)
简介 几乎没有单纯之考察队列的,队列一般只作为一个辅助工具 队列常服务于BFS queue接口 1.N叉树的层序遍历 link: 思路: 队列 层序遍历即可 code /* // Definition for a Node. class Node { public:int val;vector<Node*> children;Node()…...

vLLM私有化部署大语言模型LLM
目录 一、vLLM介绍 二、安装vLLM 1、安装环境 2、安装步骤 三、运行vLLM 1、运行方式 2、切换模型下载源 3、运行本地已下载模型 四、通过http访问vLLM 一、vLLM介绍 vLLM(官方网址:https://www.vllm.ai)是一种用于大规模语言模型&#x…...

OpenAI Whisper:语音识别技术的革新者—深入架构与参数
当下语音识别技术正以前所未有的速度发展,极大地推动了人机交互的便利性和效率。OpenAI的Whisper系统无疑是这一领域的佼佼者,它凭借其卓越的性能、广泛的适用性和创新的技术架构,正在重新定义语音转文本技术的规则。今天我们一起了解一下Whi…...

基于当前最前沿的前端(Vue3 + Vite + Antdv)和后台(Spring boot)实现的低代码开发平台
项目是一个基于当前最前沿的前端技术栈(Vue3 Vite Ant Design Vue,简称Antdv)和后台技术栈(Spring Boot)实现的低代码开发平台。以下是对该项目的详细介绍: 一、项目概述 项目名称:lowcode-s…...

【Rust】错误处理机制
目录 思维导图 引言 一、错误处理的重要性 1.1 软件中的错误普遍存在 1.2 编译时错误处理要求 二、错误的分类 2.1 可恢复错误(Recoverable Errors) 2.2 不可恢复错误(Unrecoverable Errors) 三、Rust 的错误处理机制 3…...

Logback日志技术
Logback日志技术 日志 日志(Logging)是软件开发和运维中用于记录系统或应用程序运行期间发生的运行信息、状态变化、错误信息等的一种机制,这种记录的方式就好像我们日常生活中写日记一样。它提供了一种持久化的方式,使得开发者…...
9分布式微服务架构
分布式微服务架构不光需要从架构上的设计优化系统,还要在编码上优化达到最好的效果 中心化的设计 中心化的设计比较简单,分布式集群中的角色分为两种,管理者和被管理者。 在一个分布式或者集群中,管理者角色管理着其他处理实际…...

Leecode刷题C语言之统计重新排列后包含另一个字符串的子字符串数目②
执行结果:通过 执行用时和内存消耗如下: void update(int *diff, int c, int add, int *cnt) {diff[c] add;if (add 1 && diff[c] 0) {// 表明 diff[c] 由 -1 变为 0(*cnt)--;} else if (add -1 && diff[c] -1) {// 表明 diff[c] 由 0 变为 -…...

Redis相关知识总结(缓存雪崩,缓存穿透,缓存击穿,Redis实现分布式锁,如何保持数据库和缓存一致)
文章目录 1.什么是Redis?2.为什么要使用redis作为mysql的缓存?3.什么是缓存雪崩、缓存穿透、缓存击穿?3.1缓存雪崩3.1.1 大量缓存同时过期3.1.2 Redis宕机 3.2 缓存击穿3.3 缓存穿透3.4 总结 4. 数据库和缓存如何保持一致性5. Redis实现分布式…...

Debian系统简介
目录 Debian系统介绍 Debian版本介绍 Debian软件源介绍 软件包管理工具dpkg dpkg核心指令详解 安装软件包 卸载软件包 查询软件包状态 验证软件包完整性 手动处理依赖关系 dpkg vs apt Debian系统介绍 Debian 和 Ubuntu 都是基于 Debian内核 的 Linux 发行版ÿ…...
电脑插入多块移动硬盘后经常出现卡顿和蓝屏
当电脑在插入多块移动硬盘后频繁出现卡顿和蓝屏问题时,可能涉及硬件资源冲突、驱动兼容性、供电不足或系统设置等多方面原因。以下是逐步排查和解决方案: 1. 检查电源供电问题 问题原因:多块移动硬盘同时运行可能导致USB接口供电不足&#x…...
工程地质软件市场:发展现状、趋势与策略建议
一、引言 在工程建设领域,准确把握地质条件是确保项目顺利推进和安全运营的关键。工程地质软件作为处理、分析、模拟和展示工程地质数据的重要工具,正发挥着日益重要的作用。它凭借强大的数据处理能力、三维建模功能、空间分析工具和可视化展示手段&…...
今日科技热点速览
🔥 今日科技热点速览 🎮 任天堂Switch 2 正式发售 任天堂新一代游戏主机 Switch 2 今日正式上线发售,主打更强图形性能与沉浸式体验,支持多模态交互,受到全球玩家热捧 。 🤖 人工智能持续突破 DeepSeek-R1&…...

c#开发AI模型对话
AI模型 前面已经介绍了一般AI模型本地部署,直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型,但是目前国内可能使用不多,至少实践例子很少看见。开发训练模型就不介绍了&am…...
JVM暂停(Stop-The-World,STW)的原因分类及对应排查方案
JVM暂停(Stop-The-World,STW)的完整原因分类及对应排查方案,结合JVM运行机制和常见故障场景整理而成: 一、GC相关暂停 1. 安全点(Safepoint)阻塞 现象:JVM暂停但无GC日志,日志显示No GCs detected。原因:JVM等待所有线程进入安全点(如…...
#Uniapp篇:chrome调试unapp适配
chrome调试设备----使用Android模拟机开发调试移动端页面 Chrome://inspect/#devices MuMu模拟器Edge浏览器:Android原生APP嵌入的H5页面元素定位 chrome://inspect/#devices uniapp单位适配 根路径下 postcss.config.js 需要装这些插件 “postcss”: “^8.5.…...

【C++特殊工具与技术】优化内存分配(一):C++中的内存分配
目录 一、C 内存的基本概念 1.1 内存的物理与逻辑结构 1.2 C 程序的内存区域划分 二、栈内存分配 2.1 栈内存的特点 2.2 栈内存分配示例 三、堆内存分配 3.1 new和delete操作符 4.2 内存泄漏与悬空指针问题 4.3 new和delete的重载 四、智能指针…...

【JVM面试篇】高频八股汇总——类加载和类加载器
目录 1. 讲一下类加载过程? 2. Java创建对象的过程? 3. 对象的生命周期? 4. 类加载器有哪些? 5. 双亲委派模型的作用(好处)? 6. 讲一下类的加载和双亲委派原则? 7. 双亲委派模…...