golang builder,buffer,+的对比

这篇博客探讨了在Golang中使用Buffer和Builder与使用+或+=操作符拼接字符串的性能差异。通过基准测试显示,Buffer和Builder在性能上优于+操作,其中Builder表现最佳。Buffer在拼接时会先判断并扩容,而+操作则涉及额外的数组转换和长度计算步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  • 使用buffer或是builder来拼接字符串,会比使用+或+=快
func BenchmarkBuffer(b *testing.B) {
	text := "test"
	b.ResetTimer()
	buffer := bytes.Buffer{}
	for i := 0; i < b.N; i++ {
		buffer.WriteString(text)
	}
	buffer.String()
}

func BenchmarkBuilder(b *testing.B) {
	text := "test"
	b.ResetTimer()
	builder := strings.Builder{}
	for i := 0; i < b.N; i++ {
		builder.WriteString(text)
	}
	builder.String()
}

func BenchmarkString(b *testing.B) {
	text := "test"
	b.ResetTimer()
	for i := 0; i < b.N; i++ {
		text += "test"
	}
}

结果

goos: windows
goarch: amd64
pkg: test/one
BenchmarkBuffer
BenchmarkBuffer-12         181560793             8.03 ns/op
BenchmarkBuilder
BenchmarkBuilder-12        360210758             3.86 ns/op
BenchmarkString
BenchmarkString-12           353895         96676 ns/op

func BenchmarkBuffer(b *testing.B) {
	text := ""
	b.ResetTimer()
	for i := 0; i < b.N; i++ {
		buffer := bytes.Buffer{}
		buffer.WriteString(text)
		buffer.WriteString(text)
		buffer.String()
	}
}

func BenchmarkBuilder(b *testing.B) {
	text := ""
	b.ResetTimer()
	for i := 0; i < b.N; i++ {
		builder := strings.Builder{}
		builder.WriteString(text)
		builder.WriteString(text)
		builder.String()
	}
}

func BenchmarkString(b *testing.B) {
	text := ""
	b.ResetTimer()
	for i := 0; i < b.N; i++ {
		text = ""
		text += ""
	}
}

结果

goos: windows
goarch: amd64
pkg: test/one
BenchmarkBuffer
BenchmarkBuffer-12         136879110             8.73 ns/op
BenchmarkBuilder
BenchmarkBuilder-12        499198370             2.44 ns/op
BenchmarkString
BenchmarkString-12         276599348             4.29 ns/op

第二种空字符串的可能在应用上基本不会用到,通常情况下我们用builder和buffer会比+快很多。

func (b *Builder) WriteString(s string) (int, error) {
	b.copyCheck()
	b.buf = append(b.buf, s...)
	return len(s), nil
}

builder的字符串相加就是直接用的append

func (b *Buffer) WriteString(s string) (n int, err error) {
	b.lastRead = opInvalid
	m, ok := b.tryGrowByReslice(len(s))
	if !ok {
		m = b.grow(len(s))
	}
	return copy(b.buf[m:], s), nil
}

buffer的字符串相加用的是copy,不同的是它会先去判断现有buf够不够长,不够的话会进行2倍的扩容

func concatstrings(buf *tmpBuf, a []string) string {
	idx := 0
	l := 0
	count := 0
	for i, x := range a {
		n := len(x)
		if n == 0 {
			continue
		}
		if l+n < l {
			throw("string concatenation too long")
		}
		l += n
		count++
		idx = i
	}
	if count == 0 {
		return ""
	}

	// If there is just one string and either it is not on the stack
	// or our result does not escape the calling frame (buf != nil),
	// then we can return that string directly.
	if count == 1 && (buf != nil || !stringDataOnStack(a[idx])) {
		return a[idx]
	}
	s, b := rawstringtmp(buf, l)
	for _, x := range a {
		copy(b, x)
		b = b[len(x):]
	}
	return s
}

而+号则是先把相加的字符串变成数组再设置总长度,最后才是拷贝

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值