淺析Golang中字符串拼接問題
1.概述
Go的字符串是一個不可改變的數(shù)據(jù)結(jié)構(gòu),這和其他語言如JAVA,C++等的設定很類似.總體來說,有如下五種拼接方式,下面我們將論述各種方式的性能問題,以及如何選擇.
(golang字符串,內(nèi)存模型)
type StringHeader struct { Data uintptr Len int }
注意:字符串具有不可改變的特性,即便通過指針等變相操作
var a string = "old" bptr := (*reflect.StringHeader)(unsafe.Pointer(&a)) dataPtr := (*byte)(unsafe.Pointer(bptr.Data)) var b = [3]byte{'n', 'e', 'w'} *dataPtr = b[0] //報錯 fmt.Println(bptr)
2.Golang中字符串拼接的方式
方式一、直接+
當使用連接符 + 拼接兩個字符串時,會生成一個新的字符串并開辟新的內(nèi)存空間,空間大小等于兩個字符串之和。在訓中中時,不斷拼接新的字符串,這樣就會不斷申請內(nèi)存空間, 性能就會越來越差。 所以,在字符串密集拼接場景中,使用 + 會嚴重降低性能。包括熱路徑的代碼.
方式二、strings.Builder
func Benchmark_StringsBuilder(b *testing.B) { var sb strings.Builder for i := 0; i < b.N; i++ { sb.WriteString("hello world") } _ = sb.String() }
方式三、bytes.Buffer
func Benchmark_BytesBuffer(b *testing.B) { var buf bytes.Buffer for i := 0; i < b.N; i++ { buf.WriteString("hello world") } _ = buf.String() }
方式四、fmt.Fprint(&buf,&str)
方式五、strings.Join
性能不是最優(yōu),但在切片的情況下,可以用來拼接
3.總結(jié)
Benchmark_StringAdd Benchmark_StringAdd-8 117806 127059 ns/op Benchmark_BytesBuffer Benchmark_BytesBuffer-8 38938282 25.88 ns/op Benchmark_StringsBuilder Benchmark_StringsBuilder-8 57249450 18.53 ns/op
3.1 性能方面,strings.Builder 比 bytes.Buffer 快差不多 20%,
原因:strings.Builder 和 bytes.Buffer 底層都是一個 []byte,但是 bytes.Buffer 轉(zhuǎn)換字符串時會重新申請內(nèi)存空間用來存放, 而 strings.Builder 直接將底層的 []byte 利用指針的方式強轉(zhuǎn)為字符串.
//strings.Builder的String() func (b *Builder) String() string { return *(*string)(unsafe.Pointer(&b.buf)) } //bytes.Builder的String() func (b *Buffer) String() string { if b == nil { // Special case, useful in debugging. return "<nil>" } return string(b.buf[b.off:]) }
3.2 strings.Builder通常性能最優(yōu),但底層依賴于[]byte,所以如果平凡擴容就不妙了,因此我們需要借助它的Grow方法,以已分配最終[]byte的容量,避免因為擴容帶來的性能損失
func Benchmark_StringConcat(b *testing.B) { str := "hello world" var sb strings.Builder sb.Grow(b.N * len(str)) for i := 0; i < b.N; i++ { sb.WriteString(str) } _ = sb.String() }
3.3 strings.Builder沒有拷貝構(gòu)造(借用C++說法),因為
type Builder struct { addr *Builder // of receiver, to detect copies by value buf []byte //如果拷貝,這個buf共享,最后導致數(shù)據(jù)混亂 }
到此這篇關(guān)于淺析Golang中字符串拼接問題的文章就介紹到這了,更多相關(guān)Golang字符串拼接內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語言kube-scheduler深度剖析與開發(fā)之pod調(diào)度
這篇文章主要為大家介紹了Go語言kube-scheduler深度剖析與開發(fā),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-04-04Go語言實現(xiàn)Fibonacci數(shù)列的方法
這篇文章主要介紹了Go語言實現(xiàn)Fibonacci數(shù)列的方法,實例分析了使用遞歸和不使用遞歸兩種技巧,并對算法的效率進行了對比,需要的朋友可以參考下2015-02-02在Golang中正確的修改HTTPRequest的Host的操作方法
我們工作中經(jīng)常需要通過HTTP請求Server的服務,比如腳本批量請求接口跑數(shù)據(jù),由于一些網(wǎng)關(guān)策略,部分Server會要求請求中Header里面附帶Host參數(shù),所以本文給大家介紹了如何在Golang中正確的修改HTTPRequest的Host,需要的朋友可以參考下2023-12-12Go單元測試對數(shù)據(jù)庫CRUD進行Mock測試
這篇文章主要為大家介紹了Go單元測試對數(shù)據(jù)庫CRUD進行Mock測試的示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-06-06