Go 标准库log 包需通过 io.MultiWriter 实现终端与文件双写入,并用 %+v 格式化错误以保留堆栈;可封装带前缀的 logger 模拟级别,大规模项目推荐 zap 或 zerolog 等结构化日志库。
Go 语言标准库的 log 包本身不直接支持错误类型自动展开,但通过合理封装和组合,可以实现「错误信息 + 上下文日志」同时输出到文件和终端,并保持结构清晰、可读性强。
Go 的 log.Logger 支持自定义输出目标(io.Writer)。你可以用 io.MultiWriter 将日志同时发给 os.Stdout 和文件句柄:
os.OpenFile 配合 os.O_APPEND | os.O_CREATE | os.O_WRONLY)io.MultiWriter,传入 os.Stdout 和文件句柄log.SetOutput 或新建 log.New 实例绑定该 writer这样所有 log.Print/Printf/Println 调用都会同步出现在终端和文件中。
标准 log 打印 error 时默认只调用 err.Error(),丢失堆栈。推荐两种轻量方案:
fmt.Printf("%+v", err):若使用 github.com/pkg/errors 或 Go 1.13+ 的 errors.Unwrap / %w,%+v 可显示完整调用链func LogError(msg string, err error) {
log.Printf("%s: %+v", msg, err)
}标准 log 没有内置级别,但可通过前缀模拟(如 [ERROR]、[WARN]):
log.New(mw, "[ERROR] ", log.LstdFlags)
errorLogger.Println(...),正常流程用 infoLogger
log.Fatal(会自动加换行并退出),或手动 os.Exit(1)
注意:不要把非致命错误都塞进 Fatal,否则掩盖真正需终止的场景。
如果项目规模上升,建议切换到结构化日志库,比如:
logger.Error("db query failed", zap.Error(err), zap.String("query", sql))
log.Error().Err(err).Str("path", r.URL.Path).Msg("request failed")
它们天然支持错误字段序列化、JSON 输出、多写入器(console + file + network),比手写更健壮且易扩展。