用法:1、將map作為Set使用時,可以將值類型定義為空結(jié)構(gòu)體,僅作為占位符使用;channel不需要發(fā)送數(shù)據(jù),只用來通知子協(xié)程執(zhí)行任務(wù)等時,可使用空結(jié)構(gòu)體作為占位符;3、結(jié)構(gòu)體只包含方法,不包含任何的字段時,可聲明空結(jié)構(gòu)體,以節(jié)省資源。
本教程操作環(huán)境:windows10系統(tǒng)、GO 1.11.2、Dell G3電腦。
1、空結(jié)構(gòu)體占用空間么
在 Go 語言中,我們可以使用 unsafe.Sizeof 計算出一個數(shù)據(jù)類型實(shí)例需要占用的字節(jié)數(shù)。
package main import ( "fmt" "unsafe" ) func main() { fmt.Println(unsafe.Sizeof(struct{}{})) }
運(yùn)行上面的例子將會輸出:
$ go run main.go 0
也就是說,空結(jié)構(gòu)體 struct{} 實(shí)例不占據(jù)任何的內(nèi)存空間。
2、空結(jié)構(gòu)體的作用
因?yàn)榭战Y(jié)構(gòu)體不占據(jù)內(nèi)存空間,因此被廣泛作為各種場景下的占位符使用。一是節(jié)省資源,二是空結(jié)構(gòu)體本身就具備很強(qiáng)的語義,即這里不需要任何值,僅作為占位符。
2.1 實(shí)現(xiàn)集合(Set)
Go 語言標(biāo)準(zhǔn)庫沒有提供 Set 的實(shí)現(xiàn),通常使用 map 來代替。事實(shí)上,對于集合來說,只需要 map 的鍵,而不需要值。即使是將值設(shè)置為 bool 類型,也會多占據(jù) 1 個字節(jié),那假設(shè) map 中有一百萬條數(shù)據(jù),就會浪費(fèi) 1MB 的空間。
因此呢,將 map 作為集合(Set)使用時,可以將值類型定義為空結(jié)構(gòu)體,僅作為占位符使用即可。
type Set map[string]struct{} func (s Set) Has(key string) bool { _, ok := s[key] return ok } func (s Set) Add(key string) { s[key] = struct{}{} } func (s Set) Delete(key string) { delete(s, key) } func main() { s := make(Set) s.Add("Tom") s.Add("Sam") fmt.Println(s.Has("Tom")) fmt.Println(s.Has("Jack")) }
2.2 不發(fā)送數(shù)據(jù)的信道(channel)
func worker(ch chan struct{}) { <-ch fmt.Println("do something") close(ch) } func main() { ch := make(chan struct{}) go worker(ch) ch <- struct{}{} }
有時候使用 channel 不需要發(fā)送任何的數(shù)據(jù),只用來通知子協(xié)程(goroutine)執(zhí)行任務(wù),或只用來控制協(xié)程并發(fā)度。這種情況下,使用空結(jié)構(gòu)體作為占位符就非常合適了。
2.3 僅包含方法的結(jié)構(gòu)體
type Door struct{} func (d Door) Open() { fmt.Println("Open the door") } func (d Door) Close() { fmt.Println("Close the door") }
在部分場景下,結(jié)構(gòu)體只包含方法,不包含任何的字段。例如上面例子中的 Door,在這種情況下,Door 事實(shí)上可以用任何的數(shù)據(jù)結(jié)構(gòu)替代。例如:
type Door int type Door bool
無論是 int 還是 bool 都會浪費(fèi)額外的內(nèi)存,因此呢,這種情況下,聲明為空結(jié)構(gòu)體是最合適的。
推薦學(xué)習(xí):Golang教程