strings.Builder 是 Go 中最高效的字符串拼接方式,因其预分配字节数组、避免重复扩容与拷贝,降低 GC 压力;而 + 拼接因字符串不可变导致 O(n²) 时间复杂度和多次堆分配。

使用 strings.Builder 拼接字符串是 Go 中最高效的方式之一,它通过预分配底层字节数组、避免重复扩容和内存拷贝,显著减少 GC 压力和分配次数。
为什么不用 + 或 += 拼接?
Go 中字符串是不可变的。每次用 + 拼接都会创建新字符串,底层复制所有字节。拼接 n 次,时间复杂度接近 O(n²),且触发多次堆分配。
例如:
s := ""s += "a"
s += "b"
s += "c"
实际发生了 3 次内存分配和至少 3 次字节拷贝(长度递增:1→2→3)。
立即学习“go语言免费学习笔记(深入)”;
strings.Builder 的核心优势
strings.Builder 内部持有一个可增长的 []byte,写入时直接追加,不涉及字符串转换开销(直到调用 String() 才生成最终字符串)。它还实现了 io.Writer 接口,兼容各类写入逻辑。
- 零拷贝追加:只要容量足够,
Write/WriteString直接写入底层数组 - 可控预分配:用
Grow(n)预留空间,避免中间扩容 - 无额外字符串分配:构造过程中不生成中间字符串
- 安全复用:调用
Reset()可清空并重用,适合循环场景
典型高效用法示例
正确姿势:
var b strings.Builder// 预估总长,提前分配(可选但推荐)
b.Grow(1024)
b.WriteString("Hello")
b.WriteByte(' ')
b.WriteString("World")
b.WriteRune('!')
result := b.String() // 仅此处一次分配
批量拼接循环中复用 Builder:
var b strings.Builderfor i, v := range values {
if i > 0 { b.WriteByte(',') }
b.WriteString(v)
}
return b.String()
若需多次生成不同结果,用 b.Reset() 清空而非重新声明变量。
注意事项与避坑点
-
Builder不是线程安全的,多 goroutine 写入需加锁或各自实例化 - 不要对 Builder 取地址后传参再修改——它的零值有效,但复制后原实例状态不共享
-
String()返回后,Builder 底层数组可能被复用;若需长期持有结果字符串,无需额外操作(Go 字符串是只读的,底层字节不会被 Builder 修改) - 小量拼接(如 2–3 个短字符串)用
+也无明显性能差异,不必过度优化










