空结构体的使用
何为空结构体
在结构体中,可以包裹一系列与对象相关的属性,但若该对象没有属性呢?那它就是一个空结构体。
空结构体,和正常的结构体一样,可以接收方法函数。
eg:
type Lamp struct{}func (l Lamp) On() {println("On")}
func (l Lamp) Off() {println("Off")
}
空结构体的作用
因为空结构体不占据内存空间
,因此被广泛作为各种场景下的占位符使用。一是节省资源,二是空结构体本身就具备很强的语义,即这里不需要任何值,仅作为占位符。
1.实现集合(set)
Go 语言标准库没有提供 Set 的实现,通常使用 map 来代替。事实上,对于集合来说,只需要 map 的键,而不需要值。即使是将值设置为 bool 类型,也会多占据 1 个字节,那假设 map 中有一百万条数据,就会浪费 1MB 的空间。
因此呢,将map作为集合(Set)使用时,可以将值类型定义为空结构体,仅作为占位符使用即可。
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("lisan")s.Add("sam")fmt.Println(s.Has("lisan"))s.Delete("lisan")fmt.Println(s.Has("lisan"))/* truefalse */
}
不发送数据的信道(channel)
func worker(ch chan struct{}) {<-chfmt.Println("do something")close(ch)
}func main() {ch := make(chan struct{})go worker(ch)ch <- struct{}{}
}
有时候使用 channel 不需要发送任何的数据,只用来通知子协程(goroutine)执行任务,或只用来控制协程并发度。这种情况下,使用空结构体作为占位符就非常合适了。
仅包含方法的结构体
type Door struct{}func (d Door) Open() {fmt.Println("Open the door")
}func (d Door) Close() {fmt.Println("Close the door")
}
在部分场景下,结构体只包含方法,不包含任何的字段。例如上面例子中的 Door,在这种情况下,Door 事实上可以用任何的数据结构替代。例如:
type Door int
type Door bool
但无论是 int 还是 bool 都会浪费额外的内存,因此呢,这种情况下,声明为空结构体是最合适的。