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

Go 中间件学习

文章目录

      • 1. 设计中间件接口
      • 2. 创建中间件函数
      • 3. 使用中间件
      • 4. 中间件链
      • 5. 使用第三方框架
      • 6. 定义自定义中间件
      • 7. 使用自定义中间件
        • 应用到整个路由组
        • 应用到单个路由
      • 8. 访问上下文中的信息
      • 9. 控制流程
      • 总结

在 Go 中, 中间件(Middleware) 是一种设计模式,常用于在请求处理的各个阶段添加通用功能,如日志记录、身份验证、异常处理等。中间件通常应用于 HTTP 服务器(如 net/http 包或第三方框架如 Gin)中,但也可以在其他需要分层处理的场景中使用。

以下是设计和使用 Go 中间件插件的基本步骤:

1. 设计中间件接口

首先,为你的中间件定义一个通用的接口。中间件通常接收一个处理器函数,并返回一个包装后的处理器函数。

type Middleware func(http.Handler) http.Handler

2. 创建中间件函数

根据需求创建具体的中间件函数,这些函数遵循上面定义的接口。下面是一个简单的日志记录中间件的例子:

func LoggingMiddleware(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        log.Printf("Request URI: %s", r.RequestURI)
        next.ServeHTTP(w, r)
    })
}

3. 使用中间件

在实际的应用程序中,将中间件应用于 HTTP 处理器。例如:

func main() {
    // 创建基本的处理器
    handler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        w.Write([]byte("Hello, World!"))
    })

    // 应用中间件
    handlerWithMiddleware := LoggingMiddleware(handler)

    // 启动服务器
    http.ListenAndServe(":8080", handlerWithMiddleware)
}

4. 中间件链

你可以将多个中间件串联在一起,以形成一个中间件链:

func ChainMiddleware(h http.Handler, middlewares ...Middleware) http.Handler {
    for _, middleware := range middlewares {
        h = middleware(h)
    }
    return h
}

func main() {
    // 基本处理器
    handler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        w.Write([]byte("Hello, World!"))
    })

    // 创建中间件链
    chainedHandler := ChainMiddleware(handler, LoggingMiddleware)

    // 启动服务器
    http.ListenAndServe(":8080", chainedHandler)
}

5. 使用第三方框架

如果你使用像 Gin 或 Echo 这样的第三方框架,它们有自己的中间件体系,但基本理念是相同的。以 Gin 为例:

func main() {
    r := gin.Default()

    // 使用框架内置的日志中间件
    r.Use(gin.Logger())

    // 添加自定义中间件
    r.Use(func(c *gin.Context) {
        log.Printf("Before request")
        c.Next()
        log.Printf("After request")
    })

    // 路由处理
    r.GET("/", func(c *gin.Context) {
        c.String(http.StatusOK, "Hello, World!")
    })

    // 启动服务器
    r.Run(":8080")
}

在 Gin 框架中,自定义中间件的使用非常简单。中间件本质上是一个函数,它可以在请求处理的各个阶段执行一些通用的任务,比如日志记录、身份验证等。下面是如何创建和使用自定义中间件的步骤:

6. 定义自定义中间件

自定义中间件函数的签名如下:

func MyMiddleware() gin.HandlerFunc {
    return func(c *gin.Context) {
        // 在处理请求之前执行的操作
        log.Println("Before request")

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

        // 在处理请求之后执行的操作
        log.Println("After request")
    }
}

在这个例子中,中间件函数 MyMiddleware 返回一个 gin.HandlerFunc,它包含了在请求处理前后执行的代码。

7. 使用自定义中间件

你可以将中间件应用于整个路由组或者单个路由:

应用到整个路由组
func main() {
    // 创建一个 Gin 引擎实例
    r := gin.Default()

    // 全局使用自定义中间件
    r.Use(MyMiddleware())

    // 定义路由组
    v1 := r.Group("/v1")
    {
        v1.GET("/test", func(c *gin.Context) {
            c.String(http.StatusOK, "Test v1")
        })
    }

    // 启动服务器
    r.Run(":8080")
}

在这个例子中,MyMiddleware 中间件将应用于 /v1 路由组中的所有路由。

应用到单个路由
func main() {
    r := gin.Default()

    // 单独给某个路由使用自定义中间件
    r.GET("/test", MyMiddleware(), func(c *gin.Context) {
        c.String(http.StatusOK, "Test route")
    })

    r.Run(":8080")
}

在这里,中间件 MyMiddleware 仅应用于 /test 路由。

8. 访问上下文中的信息

在中间件中,你可以通过 c *gin.Context 访问请求和响应相关的所有信息,比如请求头、参数等。你还可以在中间件中设置一些值传递给后续的处理函数:

func MyMiddleware() gin.HandlerFunc {
    return func(c *gin.Context) {
        // 例如:在上下文中设置一个值
        c.Set("example", "some value")

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

        // 在响应返回之前执行的操作
    }
}

func main() {
    r := gin.Default()

    r.GET("/test", MyMiddleware(), func(c *gin.Context) {
        // 获取在中间件中设置的值
        example := c.MustGet("example").(string)
        c.String(http.StatusOK, example)
    })

    r.Run(":8080")
}

9. 控制流程

在中间件中,可以使用 c.Abort() 来中止请求处理,并立即返回响应。

func AuthMiddleware() gin.HandlerFunc {
    return func(c *gin.Context) {
        token := c.GetHeader("Authorization")

        if token != "expected-token" {
            c.AbortWithStatusJSON(http.StatusUnauthorized, gin.H{"error": "unauthorized"})
            return
        }

        // 如果验证通过,继续处理请求
        c.Next()
    }
}

func main() {
    r := gin.Default()

    r.GET("/secure", AuthMiddleware(), func(c *gin.Context) {
        c.String(http.StatusOK, "You have access!")
    })

    r.Run(":8080")
}

在这个例子中,如果请求头中的 Authorization 不符合预期,AuthMiddleware 将会中止请求处理,并返回 401 Unauthorized 响应。

总结

通过自定义中间件,你可以在 Gin 应用中轻松实现如日志记录、身份验证等通用功能。自定义中间件可以应用于整个路由组或单个路由,且中间件可以在请求处理的不同阶段执行逻辑,使你的应用程序更加灵活和易于维护。


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

相关文章:

  • rhcsa练习(3)
  • FreePBX 17 on ubuntu24 with Asterisk 20
  • GO随记:不使用主键id 如何分表与mysql大表
  • 通过 route 或 ip route 管理Linux主机路由
  • Personal APP
  • 【C++】B2106 矩阵转置
  • 【CanMV K230】边缘检测
  • 羊大师:白露养生经,羊奶不可少
  • docker实战基础一 (Docker基础命令)
  • 【云原生】Helm来管理Kubernetes集群的详细使用方法与综合应用实战
  • C++实现简易俄罗斯方块小游戏
  • Apache SeaTunnel Zeta 引擎源码解析(一)Server端的初始化
  • 心觉:如何填平想象和愿望之间的鸿沟?
  • Docker设置socks5代理
  • 大模型超详细解读汇总
  • 一个“改造”的工厂背后:中国电商的AI重构
  • 赛码网牛客在acm模式下利用node.js处理多行输入方法
  • 视频结构化从入门到精通——图像算法类型介绍
  • LuaJit分析(三)luajit字节码文件格式
  • 【C++】string的模拟实现
  • 1119 Pre- and Post-order Traversals
  • Django学习(二)
  • 基因对应身体的需求 平衡饮食的重要性 第四篇
  • 8个优质视频素材库,商用无忧
  • AT+MQTT指令连接华为云实现数据上传
  • 使用linux命令导出mysql数据为CSV文件