2 回答

TA貢獻(xiàn)1831條經(jīng)驗(yàn) 獲得超9個(gè)贊
不,這個(gè)例子不會(huì)導(dǎo)致內(nèi)存泄漏,因?yàn)槊看握{(diào)用push. 有時(shí)可能會(huì)保留一些字節(jié)以減少分配,但它的工作原理是一個(gè)不應(yīng)考慮的實(shí)現(xiàn)細(xì)節(jié)。
如果您正在考慮在分配切片操作的結(jié)果但從不追加時(shí)可能出現(xiàn)的類似情況。只要您了解切片的語(yǔ)義,就不會(huì)出現(xiàn)任何泄漏。
s := make([]byte, 1024)
s = s[1000:]
fmt.Println(s, len(s), cap(s))
此示例將保留前 1000 個(gè)字節(jié)已分配,但無(wú)法訪問。答案很簡(jiǎn)單,不要那樣做。不難避免,如果確實(shí)需要確保已釋放底層數(shù)組,請(qǐng)使用copy將字節(jié)移動(dòng)到新切片。
這與字符串相同:
s = s[1020:]
// may leave the first 1000 bytes allocated
這再次很容易看到發(fā)生了什么,并避免。如果您使用大字符串,通常最好使用[]byte任何方式,這樣您可以更好地控制分配,并且可以在需要時(shí)復(fù)制字節(jié)。

TA貢獻(xiàn)1982條經(jīng)驗(yàn) 獲得超2個(gè)贊
將切片表達(dá)式應(yīng)用于字符串中的結(jié)果p := s[i:j]
是一個(gè)字符串。據(jù)我所知,Go 語(yǔ)言規(guī)范(https://golang.org/ref/spec)沒有指定p
將由與s
.
然而,在 Go 1.6 和更早版本中,一個(gè)實(shí)時(shí)引用p
將s
不會(huì)被垃圾收集。然而,這可能會(huì)在 Go 的未來(lái)版本中改變。
一個(gè)有趣的事實(shí)是,Java 中的String.substring
方法在 Java 8 之前以相同的方式實(shí)現(xiàn)。然而,在 Java 8 中substring
返回一個(gè)副本。
回到你的例子。每次調(diào)用push
函數(shù)時(shí),以下行實(shí)際上都會(huì)創(chuàng)建一個(gè)新的字符串實(shí)例:
buf += s
buf 的舊實(shí)例被垃圾收集。所以你的例子不受上述問題的影響。
- 2 回答
- 0 關(guān)注
- 374 瀏覽
添加回答
舉報(bào)