fmt.Printf直接输出到标准输出且不换行,需手动加\n;fmt.Sprintf返回格式化字符串,适用于拼接构建;二者动词一致但目标不同,类型错配会panic,忽略Sprintf返回值是常见错误。

fmt.Printf 直接输出到标准输出,适合调试和日志
fmt.Printf 把格式化后的字符串直接写到 os.Stdout,不会自动换行,也不返回字符串。常见错误是误以为它会像 Python 的 print() 那样自动加换行,结果多条输出挤在同一行。
- 必须手动加
\n换行,比如fmt.Printf("name: %s\n", name) - 不适用于拼接字符串再处理的场景——它没有返回值,无法赋值给变量
- 参数类型必须与动词严格匹配,否则运行时报 panic:
panic: Printf format %d has arg string of wrong type string - 支持的动词如
%s(字符串)、%d(十进制整数)、%v(默认格式)、%+v(结构体字段名+值)等,注意%q会加双引号并转义,%x输出小写十六进制
fmt.Sprintf 返回格式化字符串,适合构建内容
fmt.Sprintf 和 Printf 用法几乎一样,但不输出,而是返回 string 类型结果。这是拼接路径、生成 SQL、构造 HTTP 请求体等场景的核心工具。
- 返回值可直接赋值:
msg := fmt.Sprintf("user %d not found", id) - 和
Printf共享全部格式动词,行为一致,只是目标不同 - 性能上比字符串拼接(
"user " + strconv.Itoa(id) + " not found")更安全、更简洁,尤其参数多时;但高频调用需注意内存分配,可考虑strings.Builder替代 - 注意空指针:如果传入
nil接口或未初始化结构体指针,%v会输出,不是 panic,但可能掩盖逻辑问题
常见错误:混用动词与参数类型,或忽略返回值
最典型的两个坑:一是把字符串当整数用 %d,二是调用 Sprintf 却没接返回值,导致“好像没生效”。
-
fmt.Printf("%d", "hello")→ 运行时报错:fmt: %d verb requires integer argument -
fmt.Sprintf("id=%d", "123")→ 编译通过但运行 panic,因为"123"是string,不是int -
fmt.Sprintf("value: %v", x)中x是nil*interface{},输出value:,容易被当成正常值 - 忘记接收
Sprintf返回值:fmt.Sprintf("a=%v", a); fmt.Println(a)—— 前一句无效果,纯属冗余
实际组合用法:Sprintf 构建,Printf 输出,避免重复格式逻辑
当既要记录日志又要返回响应内容时,别写两遍格式串。用 Sprintf 生成一次,再分别用于输出和返回。
立即学习“go语言免费学习笔记(深入)”;
msg := fmt.Sprintf("request from %s, path: %s, status: %d", r.RemoteAddr, r.URL.Path, statusCode)
fmt.Printf("[LOG] %s\n", msg) // 输出到控制台
return msg // 返回给调用方
- 这样保证格式一致,改一处就全更新
- 如果需要带颜色或时间戳,可在
msg生成后包装,而不是塞进每个Printf - 注意
r.RemoteAddr可能含端口(如"127.0.0.1:54321"),若只需 IP,得先用strings.Split(r.RemoteAddr, ":")[0]
%v 打印结构体,但嵌套深了字段顺序、零值省略、私有字段是否显示,都依赖具体类型实现——别假设它总按你想要的方式展开。










