Nota aggiunta nel 2018
Da Go 1.10 c'è un strings.Builder
tipo, per favore dai un'occhiata a questa risposta per maggiori dettagli .
Risposta pre-201x
Il codice di riferimento di @ cd1 e altre risposte sono errati. b.N
non dovrebbe essere impostato nella funzione benchmark. Viene impostato in modo dinamico dallo strumento go test per determinare se il tempo di esecuzione del test è stabile.
Una funzione di benchmark dovrebbe eseguire gli stessi b.N
tempi di test e il test all'interno del loop dovrebbe essere lo stesso per ogni iterazione. Quindi lo aggiusto aggiungendo un loop interno. Aggiungo anche benchmark per alcune altre soluzioni:
package main
import (
"bytes"
"strings"
"testing"
)
const (
sss = "xfoasneobfasieongasbg"
cnt = 10000
)
var (
bbb = []byte(sss)
expected = strings.Repeat(sss, cnt)
)
func BenchmarkCopyPreAllocate(b *testing.B) {
var result string
for n := 0; n < b.N; n++ {
bs := make([]byte, cnt*len(sss))
bl := 0
for i := 0; i < cnt; i++ {
bl += copy(bs[bl:], sss)
}
result = string(bs)
}
b.StopTimer()
if result != expected {
b.Errorf("unexpected result; got=%s, want=%s", string(result), expected)
}
}
func BenchmarkAppendPreAllocate(b *testing.B) {
var result string
for n := 0; n < b.N; n++ {
data := make([]byte, 0, cnt*len(sss))
for i := 0; i < cnt; i++ {
data = append(data, sss...)
}
result = string(data)
}
b.StopTimer()
if result != expected {
b.Errorf("unexpected result; got=%s, want=%s", string(result), expected)
}
}
func BenchmarkBufferPreAllocate(b *testing.B) {
var result string
for n := 0; n < b.N; n++ {
buf := bytes.NewBuffer(make([]byte, 0, cnt*len(sss)))
for i := 0; i < cnt; i++ {
buf.WriteString(sss)
}
result = buf.String()
}
b.StopTimer()
if result != expected {
b.Errorf("unexpected result; got=%s, want=%s", string(result), expected)
}
}
func BenchmarkCopy(b *testing.B) {
var result string
for n := 0; n < b.N; n++ {
data := make([]byte, 0, 64) // same size as bootstrap array of bytes.Buffer
for i := 0; i < cnt; i++ {
off := len(data)
if off+len(sss) > cap(data) {
temp := make([]byte, 2*cap(data)+len(sss))
copy(temp, data)
data = temp
}
data = data[0 : off+len(sss)]
copy(data[off:], sss)
}
result = string(data)
}
b.StopTimer()
if result != expected {
b.Errorf("unexpected result; got=%s, want=%s", string(result), expected)
}
}
func BenchmarkAppend(b *testing.B) {
var result string
for n := 0; n < b.N; n++ {
data := make([]byte, 0, 64)
for i := 0; i < cnt; i++ {
data = append(data, sss...)
}
result = string(data)
}
b.StopTimer()
if result != expected {
b.Errorf("unexpected result; got=%s, want=%s", string(result), expected)
}
}
func BenchmarkBufferWrite(b *testing.B) {
var result string
for n := 0; n < b.N; n++ {
var buf bytes.Buffer
for i := 0; i < cnt; i++ {
buf.Write(bbb)
}
result = buf.String()
}
b.StopTimer()
if result != expected {
b.Errorf("unexpected result; got=%s, want=%s", string(result), expected)
}
}
func BenchmarkBufferWriteString(b *testing.B) {
var result string
for n := 0; n < b.N; n++ {
var buf bytes.Buffer
for i := 0; i < cnt; i++ {
buf.WriteString(sss)
}
result = buf.String()
}
b.StopTimer()
if result != expected {
b.Errorf("unexpected result; got=%s, want=%s", string(result), expected)
}
}
func BenchmarkConcat(b *testing.B) {
var result string
for n := 0; n < b.N; n++ {
var str string
for i := 0; i < cnt; i++ {
str += sss
}
result = str
}
b.StopTimer()
if result != expected {
b.Errorf("unexpected result; got=%s, want=%s", string(result), expected)
}
}
L'ambiente è OS X 10.11.6, Intel Core i7 a 2,2 GHz
Risultati del test:
BenchmarkCopyPreAllocate-8 20000 84208 ns/op 425984 B/op 2 allocs/op
BenchmarkAppendPreAllocate-8 10000 102859 ns/op 425984 B/op 2 allocs/op
BenchmarkBufferPreAllocate-8 10000 166407 ns/op 426096 B/op 3 allocs/op
BenchmarkCopy-8 10000 160923 ns/op 933152 B/op 13 allocs/op
BenchmarkAppend-8 10000 175508 ns/op 1332096 B/op 24 allocs/op
BenchmarkBufferWrite-8 10000 239886 ns/op 933266 B/op 14 allocs/op
BenchmarkBufferWriteString-8 10000 236432 ns/op 933266 B/op 14 allocs/op
BenchmarkConcat-8 10 105603419 ns/op 1086685168 B/op 10000 allocs/op
Conclusione:
CopyPreAllocate
è il modo più veloce; AppendPreAllocate
è abbastanza vicino a No.1, ma è più facile scrivere il codice.
Concat
ha prestazioni pessime sia per la velocità che per l'utilizzo della memoria. Non usarlo.
Buffer#Write
e Buffer#WriteString
sono sostanzialmente gli stessi in termini di velocità, contrariamente a quanto ha detto @ Dani-Br nel commento. Considerare che string
è davvero []byte
in Go, ha senso.
- bytes.Buffer utilizza fondamentalmente la stessa soluzione
Copy
con la tenuta di libri extra e altre cose.
Copy
e Append
usa una dimensione bootstrap di 64, uguale a byte.Buffer
Append
usa più memoria e alloc, penso che sia correlato all'algoritmo di crescita che usa. Non sta crescendo la memoria velocemente quanto i byte. Buffer
Suggerimento:
- Per un compito semplice come quello che vuole OP, vorrei usare
Append
o AppendPreAllocate
. È abbastanza veloce e facile da usare.
- Se è necessario leggere e scrivere il buffer contemporaneamente, utilizzare
bytes.Buffer
ovviamente. È per questo che è progettato.