您好,登錄后才能下訂單哦!
字符串拼接在 golang 里面其實有很多種實現(xiàn)。
實現(xiàn)方式
直接使用運算符
func BenchmarkAddStringWithOperator(b *testing.B) {
hello := "hello"
world := "world"
for i := 0; i < b.N; i++ {
_ = hello + "," + world
}
}
golang里面的字符串都是不可變的,每次運算都會產(chǎn)生一個新的字符串,所以會產(chǎn)生很多臨時的無用的字符串,不僅沒有用,還會給gc帶來額外的負擔(dān),所以性能比較差
fmt.Sprintf()
func BenchmarkAddStringWithSprintf(b *testing.B) {
hello := "hello"
world := "world"
for i := 0; i < b.N; i++ {
_ = fmt.Sprintf("%s,%s", hello, world)
}
}
內(nèi)部使用[]byte實現(xiàn),不像直接運算符這種會產(chǎn)生很多臨時的字符串,但是內(nèi)部的邏輯比較復(fù)雜,有很多額外的判斷,還用到了interface,所以性能也不是很好
strings.Join()
func BenchmarkAddStringWithJoin(b *testing.B) {
hello := "hello"
world := "world"
for i := 0; i < b.N; i++ {
_ = strings.Join([]string{hello, world}, ",")
}
}
join會根據(jù)字符串?dāng)?shù)組的內(nèi)容,計算出一個拼接之后的長度,然后申請對應(yīng)大小的內(nèi)存,一個一個字符填入,在已有一個數(shù)組的情況下,這種效率會很高,但是本來沒有,去構(gòu)造這個數(shù)據(jù)的代價也不小
buffer.WriteString()
func BenchmarkAddStringWithBuffer(b *testing.B) {
hello := "hello"
world := "world"
for i := 0; i < 1000; i++ {
var buffer bytes.Buffer
buffer.WriteString(hello)
buffer.WriteString(",")
buffer.WriteString(world)
_ = buffer.String()
}
}
這個比較理想,可以當(dāng)成可變字符使用,對內(nèi)存的增長也有優(yōu)化,如果能預(yù)估字符串的長度,還可以用buffer.Grow()接口來設(shè)置capacity
測試結(jié)果
BenchmarkAddStringWithOperator-8 50000000 30.3 ns/op
BenchmarkAddStringWithSprintf-8 5000000 261 ns/op
BenchmarkAddStringWithJoin-8 30000000 58.7 ns/op
BenchmarkAddStringWithBuffer-8 2000000000 0.00 ns/op
主要結(jié)論
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。