介绍

在平时做项目得时候,经常会看到很多包里面定义了结构体。
e.g. 在context包里面Context接口中的Done()方法,Done()返回一个是以空结构体定义数据的通道chan struct{},那这里他是有什么特殊用意吗?我们接下来分析struct{}的作用。

struct{} 空结构是没有字段的结构类型。下面看示例:

type Demo struct{}var demo struct{}

上面定义了一个Demo的结构体,var 声明变量demo结构体。

width

在介绍空结构前,先来讨论一下这个width
这里说的这个“width”来自于 gc 编译器,它描述类型实例占用的存储字节数(一个字节=8位)。width是类型的一个属性。由于 Go 程序中的每个值都有一个类型,因此该值的width由其类型定义,并且始终是 8 位的倍数

我们可以使用unsafe.Sizeof()函数获取任何值的width:

var num intvar str stringfmt.Println(unsafe.Sizeof(num)) // output: 8 fmt.Println(unsafe.Sizeof(str)) // output: 16

数组类型的width是其元素类型的倍数:

var arr [3]uint32fmt.Println(unsafe.Sizeof(arr))

结构体提供了一种更灵活的方式来定义复合类型,其width是组成类型的width加上填充的总和:

type Test struct {         num16 uint16         num32 uint32 } var test Test fmt.Println(unsafe.Sizeof(test)) // 打印 8,而不是 6

打印结果是8,这个就和内存对齐有关系了,这里不作详细的说明。

struct{}:

空结构体的字节占用是多少呢,通过下面的例子打印得到结果:

var s struct{} fmt.Println(unsafe.Sizeof(s)) // 打印 0

由于空结构体占用 零字节,因此它不需要任何的填充。所以,struct{}空结构组成的结构也不会占用任何存储空间。

type S struct {         A struct{}         B struct{} } var s S fmt.Println(unsafe.Sizeof(s)) // 打印 0

使用场景

  • 可以声明一个数组structs{},它们当然也是一样不会占用存储空间。
var arr [100000]struct{} fmt.Println(unsafe.Sizeof(arr)) // 打印 0
  • struct{}类型的切片只会占用其SliceHeader的空间:
var arr = make([]struct{}, 1000000000) fmt.Println(unsafe.Sizeof(arr)) // 打印 24

但是我们任然可以对这个切片正常操作:

var x = make([]struct{}, 100) var y = x[:50] fmt.Println(len(y), cap(y)) // 打印 50 100
  • struct{}是可寻址时,也一样可以获取它的地址:
var test struct{} var data = &test

当两个不同变量相同数据类型的时候,他们两个的地址是等的:

var a, b struct{} fmt.Println(&a == &b) // true

但是下面两个变量是不一样的:

a := make([]struct{}, 10) b := make([]struct{}, 20) fmt.Println(&a == &b) // false, a 和 b 是不同的切片fmt.Println(&a[0] == &b[0]) // true,它们的支持数组是相同的

总结

所以通过上面的分析,空结构体在内存上不占用资源,因此,我们在开发的时候可以通过chan struct{}来实现 go 协程之间信号的传递等用途。

转载:风向阅读 – Golang空结构体struct{}的作用?
地址:https://www.aiweimeng.top/archives/42.html