七叶笔记 » golang编程 » 讲讲go语言的结构体

讲讲go语言的结构体

结构体

作为C语言家族的一员,go和c一样也支持结构体。可以类比于java的一个POJO。

在学习定义结构体之前,先学习下定义一个新类型。

定义一个新类型

  type T1 int
 type T2 T1  

新类型 T1 是基于 Go 原生类型 int 定义的新自定义类型,而新类型 T2 则是 基于刚刚定义的类型 T1,定义的新类型。

这里要引入一个底层类型的概念。

如果一个新类型是基于某个 Go 原生类型定义的, 那么我们就叫 Go 原生类型为新类型的底层类型

在上面的例子中,int就是T1的底层类型。

但是T1不是T2的底层类型,只有原生类型才可以作为底层类型,所以T2的底层类型还是int

底层类型的重要性

底层类型是很重要的,因为对两个变量进行显式的类型转换,只有底层类型相同的变量间才能相互转换。底层类型是判断两个类型本质上是否相同的根本。

类型别名

  type T = string  

这种类型定义方式通常用在 项目的渐进式重构,还有对已有包的二次封装方面

类型别名表示新类型和原类型完全等价,实际上就是同一种类型。只不过名字不同而已。

定义结构体的基本形式

  // 定义结构体
 type Employee struct {
     Id   string
     Name string
     Age  int
 }  

一般我们都是定义一个有名的结构体。

字段名的大小写决定了字段是否包外可用。只有大写的字段可以被包外引用。

  // 三种初始化的方式
 func TestCreateObj(t *testing.T) {
     e := Employee{"001", "xxxx", 32} // 省略字段名。不建议
     t.Log(e)
 
     e2 := Employee{Name: "World", Age: 66}
     t.Log(e2)
 
     e3 := new(Employee) // 返回指针
     e3.Age = 111
     t.Log(e3)
 
     // 三种方式创建的对象的类型
     t.Logf("e : %T", e)   // obj_test.Employee
     t.Logf("e2 : %T", e2) // obj_test.Employee
     t.Logf("e3 : %T", e3) // *obj_test.Employee
 }  

还有一个点提一下

  e2 := Employee{Name: "World", Age: 66}  

如果换行来写

      e2 := Employee{
         Name: "World",
         Age:  66,
     }  

Age: 66,后面这个都好不能省略

还有一个点,观察e3的赋值

  e3 := new(Employee) // 返回指针
 e3.Age = 111  

new返回的是一个指针。然后指针可以直接点号赋值。这说明go默认进行了取值操作

e3.Age 等价于 (*e3).Age

空结构体

      type Empty struct {}
     var e Empty
     t.Log(unsafe.Sizeof(e))  

如上定义了一个空的结构体Empty。打印了元素e的内存大小是0。

有什么用呢?

基于空结构体类型内存零开销这样的特性,我们在日常 Go 开发中会经常使用空 结构体类型元素,作为一种“事件”信息进行 Goroutine 之间的通信

这种以空结构体为元素类建立的 channel,是目前能实现的、内存占用最小的 Goroutine 间通信方式。

 var c = make(chan Empty) // 声明一个元素类型为Empty的channel
c <- Empty{}             // 向channel写入一个“事件”  

结构体的字段可以是另一个结构体

这种形式需要说的是几个语法糖。

 type Reader struct {
ReaderName string
Age int
}

type Book struct {
BookName string
Reader Reader
}  

语法糖1:

 type Book struct {
BookName string
Reader
}  

对于结构体字段,可以省略字段名,只写结构体名。默认字段名就是结构体名

这种方式称为 嵌入字段

语法糖2:

如果是以嵌入字段形式写的结构体

 reader := Reader{"yunsheng", 20}
book := Book{"禅与摩托车维修艺术", reader}
t.Log(book.Reader.ReaderName)
t.Log(book.ReaderName)  

可以省略嵌入的Reader字段,而直接访问ReaderName

初始化问题

  1. 零值初始化 var book Book

此时book是一个各个属性全是对应类型零值的一个实例。不是nil。这种情况在Go中称为零值可用。不像java会导致npe

  1. 不建议使用字段顺序复制方式初始化如上面的代码是一个不好的示例 book := Book{“禅与摩托车维修艺术”, reader} 这样按字段顺序一个个复制的方式,问题很多:当定义的结构体字段顺序改变或者出现字段增删,必须跳转初始化的代码。或者出现非导出字段,这种方式也不支持。推荐使用“field:value”形式赋值初始化。 book := Book{BookName:”禅与摩托车维修艺术”, Reader: reader}

字段标签

结构体定义时可以在字段后面追加标签说明。

 type Employee struct {
Id   string `json:"id"`
Name string `json:"name"`
Age  int `json:"age,omitempty"`
}  

tag的格式为反单引号

 key1:"value1" key2:"value2"  

tag的作用是可以使用[反射]来检视字段的标签信息。

具体的作用还要看使用的场景。

比如这里的tag是为了帮助 encoding/json 标准包在解析对象时可以利用的规则。比如omitempty表示该字段没有值就不打印出来。

相关文章