本文首发于公众号:Hunter 后端
原文链接:Golang基础笔记五之结构体
本篇笔记介绍 Golang 中的结构体。
在 Go 中,结构体是一种用户自定义的数据类型,可以将不同类型的数据组合在一起。
以下是本篇笔记目录:
结构体的定义和使用
嵌套结构体
创建结构体递归结构
结构体标签
1、结构体的定义和使用
结构体定义的基本语法如下:
type 结构体名称 struct{ field1 type field2 type}
复制代码
我们可以先创建一个 struct:
type Person struct { Name string Age int}
复制代码
下面介绍几种方式来实例化结构体:
1. 使用结构体字面量初始化
使用结构体字面量初始化结构体变量,在创建结构体变量时,将字段值直接赋给结构体的字段。
其语法如下:
person := Person{Name: "Alice", Age: 30}fmt.Println(person)
复制代码
在这里,如果有字段没有赋值,那么这个字段的值就是该类型的零值。
2. 按字段顺序初始化
我们可以在初始化的时候不指定字段,那么它会按照结构体中的字段顺序进行赋值:
person := Person{"Alice", 30}fmt.Println(person)
复制代码
但是这个操作有个问题,就是结构体中的全部字段都需要赋值。
3. 使用 new 函数创建结构体指针
可以使用 new() 函数创建一个结构体,但是注意,其返回值是对应结构体的指针:
var person *Person = new(Person)person.Age = 1person.Name = "Hunter"
复制代码
2、嵌套结构体
结构体可以嵌套在另一个结构体里,这个就是嵌套结构体。
1. 匿名嵌套
在嵌套的时候如果直接嵌套不指定字段名,这个就叫匿名嵌套,比如下面的示例:
type Address struct { City string}type Person struct { Name string Age int Address}
复制代码
这里没有给 Address 指定字段名,所以这个 Address 就是一个匿名字段。
以下是一个初始化的例子:
person := Person{ Name: "Hunter", Age: 18, Address: Address{ City: "Beijing", },}
复制代码
使用匿名嵌套,我们可以直接访问到嵌套的结构体的字段:
这里 City 是 Addree 结构体的字段,但因为是匿名嵌套,所以可以直接访问到。
当然,也可以通过 Address 结构体进行访问:
fmt.Println(person.Address.City)
复制代码
2. 具名嵌套
如果嵌套的时候显式指定字段名,就叫具名嵌套,比如:
type Company struct { CompanyName string}type Person struct { Name string Age int CompanyInfo Company}
复制代码
这里为了以示区分,给 Company 结构体的字段名设为 CompanyInfo,其初始化的方式还是类似的:
person := Person{ Name: "Hunter", Age: 18, CompanyInfo: Company{ CompanyName: "ACompany", },}
复制代码
而如果要访问到嵌套的结构体的字段,则需要通过 CompanyInfo 进行访问:
fmt.Println(person.CompanyInfo.CompanyName)
复制代码
3、创建结构体递归结构
我们可以通过结构体嵌套自己来形成递归结构,但是嵌套的类型需要是自身结构体的指针。
下面介绍一下使用结构体来创建链表和二叉树。
1. 链表
链表的结构体如下:
type ListNode struct { Val int Next *ListNode}
复制代码
接下来创建一个链表,并打印节点内容:
head := &ListNode{Val: 1}node2 := &ListNode{Val: 2}node3 := &ListNode{Val: 3}head.Next = node2node2.Next = node3node := headfor { if node == nil { break } else { fmt.Println(node.Val) node = node.Next }}
复制代码
2. 二叉树
以下是二叉树的结构体:
type TreeNode struct { Val int Left *TreeNode Right *TreeNode}
复制代码
接下来创建一个二叉树,并用前序遍历打印二叉树节点:
func preorderTraversal(root *TreeNode) { if root == nil { return } fmt.Println(root.Val) if root.Left != nil { preorderTraversal(root.Left) } if root.Right != nil { preorderTraversal(root.Right) }}root := &TreeNode{Val: 1}root.Left = &TreeNode{Val: 2}root.Right = &TreeNode{Val: 3}root.Left.Left = &TreeNode{Val: 4}root.Left.Right = &TreeNode{Val: 5}preorderTraversal(root)
复制代码
4、结构体标签
结构体标签是附加在结构体字段后的元数据字符串,可以用于 json 序列化和反序列化、Web 框架表单数据绑定、数据库 ORM 映射等。
下面介绍一下 json 标签的使用。
json 标签
json 标签用于 JSON 的序列化和反序列化,有以下几个功能:
字段映射:json:"field_name" 指定 JSON 字段名,如果不设置,默认为 struct 的字段名
忽略字段:json:"-" 表示该字段不参与序列化
忽略空值:json:"omitempty" 表示如果字段值为零值时,则不参与序列化
类型转换:json:"field_name,string", 表示将数值转换为字符串类型
下面是一个结构体示例:
type Person struct { Name string `json:"name"` Age int `json:"age,omitempty,string"` Gender string `json:""` Address string `json:"address"` NoJsonField string `json:"-"`}
复制代码
在上面的结构体中,分别对字段实现了下面的操作:
将 Name 字段映射为 json 里的 name 字段
Age 字段映射为 age 字段,如果 Age 字段值为零值,则不参与序列化,如果 Age 字段值不为零值,则将 Age 字段值转换为字符串类型
Gender 字段映射到 json,但是不改变字段名
Address 字段映射为 address 字段
NoJsonField 字段不参与序列化
对于这些操作,我们使用下面的代码进行测试,记住,要先引入 json 模块:
import ( "encoding/json")
复制代码
然后进行操作:
person := Person{ Name: "Hunter", Age: 18, Gender: "Male", Address: "Beijing", NoJsonField: "NoJsonValue",}
_person, err := json.Marshal(person)
if err != nil { fmt.Println("JSON 编码错误:", err)} else { fmt.Println(string(_person))}
// 测试 omitempty,为零值时忽略person.Age = 0_person, err = json.Marshal(person)if err != nil { fmt.Println("JSON 编码错误:", err)} else { fmt.Println(string(_person))}
复制代码
分别打印出两条 json 数据:
{"name":"Hunter","age":"18","Gender":"Male","address":"Beijing"}{"name":"Hunter","Gender":"Male","address":"Beijing"}
复制代码
输出的结果可以印证前面我们对字段进行的定义逻辑。
除了 json 标签,还有 form 标签用于表单数据绑定,gorm 标签用于数据库映射,validate 标签用于数据验证等,后面有用到的时候再做介绍。
评论