GO错误处理【3】返回err与日志的结合

我需要在每处返回err的地方加上日志吗?

日志需要在哪个地方使用,与err如何配套使用

我现在的目录结构是pkg与 internal

pkg下面有config\infra\utils

internal 下面有cron\handler\middleware\logic\repo\model
各地方都有可能报错,日志是不是也要加在报错返回之前?日志如何使用

针对"日志谁来打"的问题,有一个最清晰的判定准则:谁负责"终结"这个错误,谁就负责打日志。

1. 为什么 repomodel 绝对不打日志?

repo(数据库操作)层,它只负责执行 SQL。如果执行失败了,它并不知道这个失败对业务意味着什么。

  • 做法:直接 return fmt.Errorf("repo: 查询用户信息失败: %w", err)
  • 理由:如果你在 repo 打印了日志,而上层 logic 发现这个错误其实可以通过读缓存补偿,从而忽略了错误,那你的日志里就会多出一条"虚假报警"。

2. 谁该打日志?(终结者们)

A. Middleware(中间件)------ 必打!

它是所有请求的最后一道防线。

  • 职责:记录所有的 HTTP 访问日志,以及 Recovery 捕获到的程序崩溃。
  • 逻辑:它拿到了最终的结果,所以它记录的是"全局视角"。

B. Cron(定时任务)------ 必打!

定时任务没有前端页面,也没有 Response。

  • 职责:如果 logic 报错返回给 cron 了,cron 就是这个错误的终结者。
  • 逻辑:它必须把错误记下来,否则没人知道后台任务失败了。

C. Handler(控制器)------ 选打(视情况而定)

  • 职责:如果一个错误 logic 处理不了,传到了 handlerhandler 准备给前端回一个 ApiError
  • 建议:如果你已经在 middleware 里记录了所有带错误信息的访问日志,那么 handler 其实可以不打,省得重复。但如果是特殊的业务警告,可以在这里打。

D. Logic(业务逻辑)------ 关键逻辑打

  • 职责:这里是处理业务的地方。
  • 场景:比如"用户支付失败",这种事关金钱的重大业务异常,即使你还没把错误返回给上层,也建议在这里打一条结构化日志,方便以后对账。

3. "报错即链条"示意图

想象一个请求的传递:

  1. Repo: 发现连接断了,return %w
  2. Logic: 收到报错,由于是关键业务,infra.Log("pay").Error("支付中断", "err", err),然后继续 return %w
  3. Handler: 收到报错,调用 res.ApiError(w, 500, ...) 返回给前端。
  4. Middleware: 最后的 Access Log 记录下:method=POST status=500 err="logic: pay fail: repo: conn broken"

结果:你的日志文件里,一条是详细的业务告警,一条是总体的 HTTP 记录。没有废话,全是干货。

相关推荐
用户67570498850210 小时前
Go 语言里判断字符串为空,90% 的人都写错了!
后端·go
用户67570498850210 小时前
Go 进阶必修:90% 的人都没用对的“表驱动法”
后端·go
知恒1 天前
Go语言接口与多态
go
知恒1 天前
Go语言变量与数据类型
go
知恒1 天前
Go包管理与模块化
go
HokKeung1 天前
飞书 lark-cli 如何存储 tenant_access_token 和 user_access_token
人工智能·go
止语Lab2 天前
sync.Pool 的真正分界线不是对象大小——一次 benchmark 翻车记录
go
HokKeung2 天前
Go 里的 IO 应该怎么管理
go
喵个咪2 天前
Go-Wind HTTP 服务器从入门到精通
后端·http·go
喵个咪2 天前
Go-Wind gRPC 服务器从入门到精通
后端·go·grpc