問個(gè) Go 問題,字符串 len == 0 和 字符串== "" ,有啥區(qū)別?
本文轉(zhuǎn)載自微信公眾號「 腦子進(jìn)煎魚了」,作者陳煎魚 。轉(zhuǎn)載本文請聯(lián)系 腦子進(jìn)煎魚了公眾號。
大家好,我是煎魚。
前幾天在微信群看到幾位大佬在討論一個(gè) Go 語言相關(guān)的問題:字符串 len == 0 和 字符串 == "",有啥區(qū)別?“
這是一個(gè)比較小的細(xì)節(jié)點(diǎn),同時(shí)也勾起了我的好奇心,因此今天這篇文章就和大家一起研究一下他們兩者有沒有區(qū)別,誰的性能更好一些?
建議在此處先自行思考一下,在心里給出自己的答案。
測試方法
在測試的方法中,我們分別聲明了 Test1 和 Test2 方法:
- func Test1() bool {
- var v string
- if v == "" {
- return true
- }
- return false
- }
- func Test2() bool {
- var v string
- if len(v) == 0 {
- return true
- }
- return false
- }
在方法內(nèi)部僅做了簡單的變量類型聲明,分別以 字符串 == "" 和 字符串 len == 0 為判斷依據(jù)。
測試用例
編寫兩個(gè)方法的 Benchmark,用于后續(xù)的性能測試:
- func BenchmarkTest1(b *testing.B) {
- for i := 0; i < b.N; i++ {
- Test1()
- }
- }
- func BenchmarkTest2(b *testing.B) {
- for i := 0; i < b.N; i++ {
- Test2()
- }
- }
結(jié)果分析
- $ go test --bench=. -benchmem
- goos: darwin
- goarch: amd64
- BenchmarkTest1-4 1000000000 0.305 ns/op 0 B/op 0 allocs/op
- BenchmarkTest2-4 1000000000 0.305 ns/op 0 B/op 0 allocs/op
- PASS
- ok _/Users/eddycjy/go-application/awesomeProject/tests 0.688s
從多次測試的結(jié)果來看,兩者比較:
性能幾乎沒有區(qū)別,甚至可以出現(xiàn)一模一樣的情況。
均不涉及內(nèi)存申請和操作,均為 0/op。說明變量并不是聲明了,就有初始化動(dòng)作的,這塊 Go 編譯器有做優(yōu)化。
結(jié)果上居然是一樣的。根據(jù)曹大的提示,我們可以進(jìn)一步看一下兩者的匯編代碼,看看具體區(qū)別在哪里:
- $ go tool compile -S main.go
- "".main STEXT nosplit size=1 args=0x0 locals=0x0
- 0x0000 00000 (main.go:3) TEXT "".main(sb), NOSPLIT|ABIInternal, $0-0
- 0x0000 00000 (main.go:3) FUNCDATA $0, gclocals·33cdeccccebe80329f1fdbee7f5874cb(sb)
- 0x0000 00000 (main.go:3) FUNCDATA $1, gclocals·33cdeccccebe80329f1fdbee7f5874cb(sb)
- 0x0000 00000 (main.go:5) RET
- 0x0000 c3 .
- go.cuinfo.packagename. SDWARFINFO dupok size=0
- 0x0000 6d 61 69 6e main
- ""..inittask SNOPTRDATA size=24
- 0x0000 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................
- 0x0010 00 00 00 00 00 00 00 00 ........
- gclocals·33cdeccccebe80329f1fdbee7f5874cb SRODATA dupok size=8
- 0x0000 01 00 00 00 00 00 00 00
無論是 len(v) == 0,又或是 v == "" 的判斷,其編譯出來的匯編代碼都是完全一致的??梢悦鞔_ Go 編譯器在這塊做了明確的優(yōu)化,大概率是直接比對了。
因此在 Go 語言中 字符串 len == 0 和 字符串 == "" 的判定,這兩者可以認(rèn)為是沒有區(qū)別的。