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

gin中间件

gin中间件

Gin 中间件(Middleware)是 Gin 框架中非常重要的概念,它允许你在请求处理流程中插入自定义逻辑。中间件可以用于日志记录、认证、权限控制、请求和响应的处理等。

中间件的基本概念

中间件是一个函数,它接收一个 gin.Context 对象作为参数,并在处理请求前后执行一些逻辑。中间件可以按顺序堆叠,形成一个中间件链。每个中间件都可以决定是否继续调用下一个中间件或直接结束请求。

中间件的常见用途

  • 日志记录:记录请求的详细信息,用于调试和性能分析。
  • 认证和授权:验证请求的认证信息,确保只有授权用户可以访问特定的资源。
  • 请求和响应处理:修改请求或响应的数据,例如添加 CORS 头、压缩响应等。
  • 错误处理:捕获和处理请求过程中出现的错误,提供统一的错误响应格式。
  • 性能监控:记录请求的处理时间,用于性能分析和优化。
  • 数据验证:在处理请求之前验证请求数据的有效性,确保数据符合预期格式。

定义中间件

在 Gin 中,中间件通常是一个函数,其签名如下:

func (c *gin.Context) {
    // 中间件逻辑
}

使用中间件

Gin 提供了几种方式来使用中间件:

  1. 全局中间件:应用于所有路由。
  2. 组中间件:应用于某个路由组。
  3. 单个路由中间件:应用于特定的路由。

示例代码

1. 全局中间件

全局中间件会在所有路由请求处理之前和之后执行。

package main

import (
	"github.com/gin-gonic/gin"
	"log"
	"time"
)

// Logger 是一个全局中间件,用于记录请求的时间、方法和路径
func Logger() gin.HandlerFunc {
	return func(c *gin.Context) {
		// 记录请求开始时间
		start := time.Now()

		// 记录请求开始的信息
		log.Println("Request started:", c.Request.Method, c.Request.URL.Path)

		// 调用下一个中间件或处理函数
		c.Next()

		// 记录请求完成的时间
		duration := time.Since(start)
		log.Println("Request completed:", c.Request.Method, c.Request.URL.Path, duration)
	}
}

func main() {
	// 创建一个新的 Gin 路由器
	r := gin.Default()

	// 注册全局中间件
	r.Use(Logger())

	// 定义一个简单的路由
	r.GET("/", func(c *gin.Context) {
		c.String(200, "Hello, World!")
	})

	// 启动 HTTP 服务器,监听 8080 端口
	r.Run(":8080")
}

2. 组中间件

组中间件应用于特定的路由组。

package main

import (
	"github.com/gin-gonic/gin"
	"log"
)

// AuthMiddleware 是一个组中间件,用于验证请求的认证信息
func AuthMiddleware() gin.HandlerFunc {
	return func(c *gin.Context) {
		// 获取请求头中的 Authorization 字段
		authToken := c.GetHeader("Authorization")

		// 检查是否有认证信息
		if authToken == "" {
			// 如果没有认证信息,返回 401 Unauthorized 错误
			c.JSON(401, gin.H{"error": "Unauthorized"})
			c.Abort() // 中断请求处理链
			return
		}

		// 继续处理请求
		c.Next()
	}
}

func main() {
	// 创建一个新的 Gin 路由器
	r := gin.Default()

	// 创建一个路由组,并注册组中间件
	protected := r.Group("/api", AuthMiddleware())
	{
		// 定义一个受保护的路由
		protected.GET("/profile", func(c *gin.Context) {
			c.JSON(200, gin.H{"message": "Profile page"})
		})
	}

	// 启动 HTTP 服务器,监听 8080 端口
	r.Run(":8080")
}

3. 单个路由中间件

单个路由中间件应用于特定的路由。

package main

import (
	"github.com/gin-gonic/gin"
	"log"
)

// LoggingMiddleware 是一个单个路由中间件,用于记录请求的信息
func LoggingMiddleware() gin.HandlerFunc {
	return func(c *gin.Context) {
		// 记录请求开始的信息
		log.Println("Logging middleware: Request started")

		// 调用下一个中间件或处理函数
		c.Next()

		// 记录请求完成的信息
		log.Println("Logging middleware: Request completed")
	}
}

func main() {
	// 创建一个新的 Gin 路由器
	r := gin.Default()

	// 定义一个带有单个路由中间件的路由
	r.GET("/ping", LoggingMiddleware(), func(c *gin.Context) {
		c.String(200, "Pong")
	})

	// 启动 HTTP 服务器,监听 8080 端口
	r.Run(":8080")
}

4.综合示例

package main

import (
	"github.com/gin-gonic/gin"
	"log"
	"time"
)

// Logger 是一个全局中间件,用于记录请求的时间、方法和路径
func Logger() gin.HandlerFunc {
	return func(c *gin.Context) {
		// 记录请求开始时间
		start := time.Now()

		// 记录请求开始的信息
		log.Println("Request started:", c.Request.Method, c.Request.URL.Path)

		// 调用下一个中间件或处理函数
		c.Next()

		// 记录请求完成的时间
		duration := time.Since(start)
		log.Println("Request completed:", c.Request.Method, c.Request.URL.Path, duration)
	}
}

// AuthMiddleware 是一个组中间件,用于验证请求的认证信息
func AuthMiddleware() gin.HandlerFunc {
	return func(c *gin.Context) {
		// 获取请求头中的 Authorization 字段
		authToken := c.GetHeader("Authorization")

		// 检查是否有认证信息
		if authToken == "" {
			// 如果没有认证信息,返回 401 Unauthorized 错误
			c.JSON(401, gin.H{"error": "Unauthorized"})
			c.Abort() // 中断请求处理链
			return
		}

		// 继续处理请求
		c.Next()
	}
}

// LoggingMiddleware 是一个单个路由中间件,用于记录请求的信息
func LoggingMiddleware() gin.HandlerFunc {
	return func(c *gin.Context) {
		// 记录请求开始的信息
		log.Println("Logging middleware: Request started")

		// 调用下一个中间件或处理函数
		c.Next()

		// 记录请求完成的信息
		log.Println("Logging middleware: Request completed")
	}
}

func main() {
	// 创建一个新的 Gin 路由器
	r := gin.Default 

总结

  1. 全局中间件:应用于所有路由,通常用于日志记录、性能监控等。
  2. 组中间件:应用于特定的路由组,通常用于认证、权限控制等。
  3. 单个路由中间件:应用于特定的路由,通常用于特定的请求处理逻辑。

http://www.kler.cn/a/378255.html

相关文章:

  • BEV数据集标注成本高?BEVPose:减少对标注数据依赖!
  • 【顶刊核心变量】上市公司企业数字创新数据(数字产品、流程、业务模式创新(2001-2023年)
  • 算法|牛客网华为机试21-30C++
  • 解决 ClickHouse 高可用集群中 VRID 冲突问题:基于 chproxy 和 keepalived 的实践分析
  • C++ 详细讲解 洛谷P1428 小鱼比可爱
  • 智能网联汽车:人工智能与汽车行业的深度融合
  • [极客大挑战 2019]FinalSQL
  • [ 问题解决篇 ] win11中本地组策略编辑器gpedit.msc打不开(gpedit.msc缺失)
  • kubectl常用命令简介
  • 万字长文详解Hive聚合函数 grouping sets、cube、rollup原理、语法、案例和优化
  • HTML 框架
  • PHP如何处理密码嗅探和重播攻击
  • Django3 + Vue.js 前后端分离书籍添加项目Web开发实战
  • 助力风力发电风机设备智能化巡检,基于YOLOv7全系列【tiny/l/x】参数模型开发构建无人机巡检场景下风机叶片缺陷问题智能化检测预警模型
  • Chrome与夸克的安全性对比
  • Vivo开奖了,劝退价。。
  • Numpy实现BatchNorm2d
  • springboot Lark扫码登录
  • WPF+MVVM案例实战(十七)- 自定义字体图标按钮的封装与实现(ABC类)
  • React v19 革新功能:2024 年需要了解的所有信息
  • 安装Go和配置镜像
  • Web Broker(Web服务应用程序)入门教程(4)
  • K3S 全面解析
  • 从0开始本地部署大模型
  • MyBatis-Plus条件构造器:构建安全、高效的数据库查询
  • NVR小程序接入平台/设备EasyNVR多个NVR同时管理视频监控新选择