0

0

如何使用Golang中间件处理请求_统一日志和认证

P粉602998670

P粉602998670

发布时间:2026-01-06 15:54:09

|

651人浏览过

|

来源于php中文网

原创

Go Web中间件通过函数链式包装实现横切关注点处理,如日志和认证;标准库用Handler装饰器模式,Gin支持全局/分组/单路由注册,需注意顺序(日志最外、panic恢复顶层)、请求ID追踪及敏感信息脱敏。

如何使用golang中间件处理请求_统一日志和认证

在 Go Web 开发中,中间件是处理横切关注点(如日志、认证、CORS、超时等)最自然的方式。使用标准 net/http 或主流框架(如 Gin、Echo、Fiber)都能轻松实现统一日志和认证逻辑,关键在于把共用逻辑“包一层”——即在真正处理业务前/后插入自定义逻辑。

用函数链式包装实现基础中间件

Go 的 http.Handlerhttp.HandlerFunc 天然支持装饰器模式。一个中间件本质上就是一个接收 http.Handler 并返回新 http.Handler 的函数:

func LoggingMiddleware(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        start := time.Now()
        log.Printf("→ %s %s", r.Method, r.URL.Path)
        next.ServeHTTP(w, r)
        log.Printf("← %s %s %v", r.Method, r.URL.Path, time.Since(start))
    })
}

func AuthMiddleware(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        auth := r.Header.Get("Authorization")
        if auth == "" || !strings.HasPrefix(auth, "Bearer ") {
            http.Error(w, "Unauthorized", http.StatusUnauthorized)
            return
        }
        token := strings.TrimPrefix(auth, "Bearer ")
        if !isValidToken(token) { // 你自己的校验逻辑
            http.Error(w, "Invalid token", http.StatusUnauthorized)
            return
        }
        next.ServeHTTP(w, r)
    })
}

然后按需组合:

handler := http.HandlerFunc(yourHandler)
handler = LoggingMiddleware(handler)
handler = AuthMiddleware(handler)
http.ListenAndServe(":8080", handler)

在 Gin 中更简洁地注册中间件

Gin 把中间件抽象为 gin.HandlerFunc,支持全局、分组、单路由三种挂载方式:

立即学习go语言免费学习笔记(深入)”;

易森网络企业版
易森网络企业版

如果您是新用户,请直接将本程序的所有文件上传在任一文件夹下,Rewrite 目录下放置了伪静态规则和筛选器,可将规则添加进IIS,即可正常使用,不用进行任何设置;(可修改图片等)默认的管理员用户名、密码和验证码都是:yeesen系统默认关闭,请上传后登陆后台点击“核心管理”里操作如下:进入“配置管理”中的&ld

下载
  • 全局中间件(所有路由生效):r.Use(Logging(), Auth())
  • 分组中间件(如只对 API 路由加认证):api := r.Group("/api").Use(Auth())
  • 单路由中间件(特殊接口跳过):r.GET("/health", HealthCheck)(不调用 Use

示例日志中间件(带请求 ID 和结构化输出):

func Logging() gin.HandlerFunc {
    return func(c *gin.Context) {
        id := uuid.New().String()
        c.Header("X-Request-ID", id)
        start := time.Now()

        c.Next() // 执行后续中间件和 handler

        latency := time.Since(start)
        status := c.Writer.Status()
        method := c.Request.Method
        path := c.Request.URL.Path

        log.Printf("[GIN] %s | %d | %v | %s %s | %s",
            id, status, latency, method, path, c.ClientIP())
    }
}

认证中间件的实用增强点

真实项目中,认证中间件不应只做“有无 token”,还需考虑:

  • 区分公开/私有接口:用路由分组或自定义属性标记(如 c.Get("skipAuth")),避免白名单硬编码
  • 解析并注入用户信息:校验通过后,把 userIDroles 等写入 c.Set("user", user),下游 handler 可直接取用
  • 支持多种认证方式:如同时接受 Bearer Token、API Key(X-API-Key)、或 Cookie(用于管理后台)
  • 错误响应标准化:统一返回 JSON 错误格式(如 {"error": "invalid_token"}),而非裸 http.Error

统一日志建议:关联请求生命周期

单纯打印时间不够,要让一次请求的所有日志可追踪:

  • context.WithValue 或 Gin 的 c.Set 传递 requestID
  • 日志库推荐 log/slog(Go 1.21+)或 zerolog,支持字段追加(如 .Str("req_id", id).Int("status", status)
  • 记录关键上下文:客户端 IP(注意反向代理场景用 X-Forwarded-For)、User-Agent、响应大小、是否命中缓存
  • 敏感字段(如密码、token)务必脱敏,日志中显示为 [REDACTED]

不复杂但容易忽略的是中间件顺序——日志通常放最外层,认证紧随其后;如果先认证再日志,失败请求可能没日志。另外,panic 恢复中间件应放在最顶层,防止崩溃中断整个链路。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

177

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

226

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

336

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

208

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

388

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

194

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

189

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

191

2025.06.17

java学习网站推荐汇总
java学习网站推荐汇总

本专题整合了java学习网站相关内容,阅读专题下面的文章了解更多详细内容。

6

2026.01.08

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.2万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号