0

0

如何使用Golang监控网络请求_Golang net/http 请求监听示例

P粉602998670

P粉602998670

发布时间:2026-01-08 08:29:02

|

566人浏览过

|

来源于php中文网

原创

Go中监听HTTP请求需实现http.Handler接口,通过中间件包装原始handler;监听请求/响应体需用io.TeeReader和自定义ResponseWriter;客户端请求监控应在http.Transport.RoundTrip层;生产环境应采样而非全量捕获,并脱敏敏感字段。

如何使用golang监控网络请求_golang net/http 请求监听示例

http.ServeMuxhttp.Handler 包裹原始 handler 实现请求监听

Go 的 net/http 没有内置“拦截器”概念,但所有请求都经过 http.Handler 接口。最直接的方式是写一个中间层 handler,在调用下游 handler 前后插入日志、计时、统计等逻辑。

常见错误是直接修改 http.DefaultServeMux 后再调用 http.ListenAndServe,却忘了自己没做任何包装——那根本不算监听,只是注册路由

  • 必须实现 http.Handler 接口(即定义 ServeHTTP(http.ResponseWriter, *http.Request) 方法)
  • 在方法体内先记录 req.URL.Pathreq.Methodreq.Header 等字段,再调用 next.ServeHTTP(w, req)
  • 别漏掉 defer 记录响应耗时:用 time.Now() 开始,time.Since() 结束
func loggingHandler(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))
    })
}

http.ListenAndServe(":8080", loggingHandler(http.DefaultServeMux))

监听请求体和响应体需用 io.TeeReader 和自定义 ResponseWriter

默认的 http.Request.Body 是单次读取流,直接 io.ReadAll(r.Body) 会导致后续 handler 读不到数据;同理,http.ResponseWriter 不允许二次写入或读取已发内容。要真正“看到”请求/响应载荷,必须做流代理。

关键点在于:不能破坏原有流语义,也不能阻塞或提前关闭连接。

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

  • 请求体监听:用 io.TeeReader(r.Body, writer)字节同时写入缓冲区和原 body,再替换 r.Body 为新 io.NopCloser 包装的缓冲读取器
  • 响应体监听:必须实现自定义 ResponseWriter,重写 Write()WriteHeader()Header() 方法,把输出先暂存到 bytes.Buffer
  • 注意状态码要在 WriteHeader() 被调用后才可读取,不能只依赖 Write()
type captureResponseWriter struct {
    http.ResponseWriter
    buf *bytes.Buffer
}

func (w *captureResponseWriter) Write(b []byte) (int, error) {
    w.buf.Write(b)
    return w.ResponseWriter.Write(b)
}

func (w *captureResponseWriter) WriteHeader(statusCode int) {
    w.ResponseWriter.WriteHeader(statusCode)
}

func captureHandler(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        var buf bytes.Buffer
        cw := &captureResponseWriter{
            ResponseWriter: w,
            buf:            &buf,
        }
        next.ServeHTTP(cw, r)
        log.Printf("response body: %s", buf.String())
    })
}

http.Transport 层监控仅适用于 HTTP 客户端出向请求

如果你要监控的是本程序作为 HTTP 客户端发出的请求(比如调用第三方 API),那得换到 http.Transport 层——它控制连接复用、超时、TLS 握手等,也能通过 RoundTrip 钩子捕获请求/响应。

DeepAI
DeepAI

为天生具有创造力的人提供的AI工具

下载

别误以为设了 http.DefaultClient.Transport 就能监听服务端进来的请求:那是两套完全不相干的路径。

  • 必须新建 &http.Transport{},并覆盖其 RoundTrip 方法
  • 在新 RoundTrip 中记录 req.URLreq.Header,再调用原 transport 的 RoundTrip
  • 响应体同样不可直接读两次,需用 io.TeeReaderhttputil.DumpResponse(仅调试,勿用于生产)
  • 注意并发安全:多个 goroutine 共享同一 transport,钩子函数里别用共享变量不加锁
originalRT := http.DefaultTransport.RoundTrip
http.DefaultTransport.RoundTrip = func(req *http.Request) (*http.Response, error) {
    log.Printf("client → %s %s", req.Method, req.URL.String())
    resp, err := originalRT(req)
    if err == nil {
        log.Printf("client ← %s %d", req.URL.String(), resp.StatusCode)
    }
    return resp, err
}

生产环境慎用全量请求体捕获,优先用结构化采样 + metrics 上报

完整记录每个请求/响应体极易拖垮性能、撑爆内存或泄露敏感字段(如 token、密码)。真实系统中,应按需采样,且只提取关键字段做指标聚合。

最容易被忽略的是 header 大小限制和 body 截断策略——不设限时,一个 100MB 的上传请求会卡住整个 handler。

  • http.MaxBytesReader 包裹 r.Body,防止恶意大 body 占用资源
  • 记录 URL、method、status、latency、user-agent、X-Request-ID 即可满足大多数可观测性需求
  • 敏感 header(如 AuthorizationCookie)必须脱敏,至少打码前几位
  • 考虑对接 OpenTelemetry:用 otelhttp.NewHandler 替代手写日志,自动注入 trace context 并导出 metrics

真要 debug 某个异常请求,靠日志 ID 查找比全局监听更高效也更安全。

相关专题

更多
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

Golang 分布式缓存与高可用架构
Golang 分布式缓存与高可用架构

本专题系统讲解 Golang 在分布式缓存与高可用系统中的应用,涵盖缓存设计原理、Redis/Etcd集成、数据一致性与过期策略、分布式锁、缓存穿透/雪崩/击穿解决方案,以及高可用架构设计。通过实战案例,帮助开发者掌握 如何使用 Go 构建稳定、高性能的分布式缓存系统,提升大型系统的响应速度与可靠性。

4

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.5万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

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

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