耐心和持久胜过激烈和狂热。
哈喽大家好,我是陈明勇,本文分享的知识是 Go 中的指针和结构体。如果本文对你有帮助,不妨点个赞,如果你是 Go 语言初学者,不妨点个关注,一起成长一起进步,如果本文有错误的地方,欢迎指出!
前言
前面的两篇文章对 Go 语言的基础语法和基本数据类型以及几个复合数据类型进行介绍,本文将对 Go 里面的指针和结构体进行介绍,也为后续文章做铺垫。
指针
在 Go
语言中,指针可以简单理解是一个地址,指针类型是依托于某一个类型而存在的,例如 Go
里面的基本数据类型 int
、float64
、string
等,它们所对应的指针类型为 *int
、*float64
、*string
等。
指针的定义
- 语法格式:
var 指针变量名 *数据类型 = &变量
。&
为取地址符号,通过&
符号获取某个变量的地址,然后赋值给指针变量。
import ( "fmt" ) func main() { var num int = 666 var numPtr *int = &num fmt.Println(numPtr) // num 变量的地址值 0xc00001c098 fmt.Println(&numPtr) // 指针变量本身的地址值 0xc00000a028 } 复制代码
npmPtr
指针变量指向变量num
,0xc00001c098
为num
变量的地址,0xc00000a028
为指针变量本身的地址值。- 使用
new(T)
函数创建指针变量
import ( "fmt" ) func main() { numPtr := new(int) fmt.Println(numPtr) // 0xc000122058 fmt.Println(*numPtr) // 0 } 复制代码
new(T)
函数为每个类型分配一片内存,内存单元保存的是对应类型的零值,函数返回一个指针类型。- 错误的类型地址赋值
func main() { var num float64 = 666 var numPtr *int = &num // cannot use &num (value of type *float64) as type *int in variable declaration } 复制代码
- 当指针所依托的类型与变量的类型不一致时,Go 编译器会报错,类型不匹配。
获取和修改指针所指向变量的值
- 通过指针获取所指向变量的值
func main() { var num int = 666 var numPtr *int = &num // 获取 num 的值 fmt.Println(*numPtr) // 666 } 复制代码
- 对指针使用
*
操作符可以获取所指向变量的值。 - 通过指针修改所指向变量的值
import ( "fmt" ) func main() { var num int = 666 var numPtr *int = &num // 修改 num 的值 *numPtr = 555 fmt.Println(num) // 555 } 复制代码
- 同时也可以对指针使用
*
操作符修改所指向变量的值。
结构体
通过上一篇文章,我们了解了数组和切片的特点,它们可以存储一组相同类型的数据,而结构体,它可以由 0 个或多个字段组成,每个字段的数据类型可以一样,也可以不一样。结构体可以表示一个对象,例如:人,人包含的一些字段有姓名、性别和年龄等。
结构体定义
语法格式:
type XXX struct { /* 结构体字段定义区域 */ } 复制代码
XXX
为结构体的名字,下面以人为对象,引入结构体
// Person定义一个人的结构体 type Person struct { // 姓名 Name string // 年龄 Age int // 性别 Sex string // 身份证号 idNumber string } 复制代码
上述代码定义了人的结构体 Person
,包含四个字段,字段的命名规则和变量是一样的,前三个字段首字母大写,可以被包外访问,第四个字段首字母小写,表示只能在包内访问。 除了上述的定义方式以外,结构体里还可以内嵌结构体
// Person 定义一个人的结构体 type Person struct { // 姓名 Name string // 年龄 Age int // 性别 Sex string // 身份证号 idNumber string } // Phone 手机结构体 type Phone struct { // 品牌 Brand string // 拥有者 Owner Person } 复制代码
上述代码定义了 Person
结构体和 Phone
结构体,Phone
结构体拥有两个字段,分别为 Brand
品牌和 Owner
拥有者,Owner
属性的类型,指定为我们所自定义的 Person
结构体。以这种方式定义的结构体字段,我们可以称为嵌入字段(Embedded Field)。也可以将这种字段称为匿名字段。
结构体的创建方式
- 1、声明一个结构体变量
// Person 定义一个人的结构体 type Person struct { // 姓名 Name string // 年龄 Age int // 性别 Sex string // 身份证号 idNumber string } func main() { var person Person fmt.Println(person.Name) // "" fmt.Println(person.Age) // 0 fmt.Println(person.Sex) // "" fmt.Println(person.idNumber) // "" // 修改结构体字段的值 person.Name = "chenmingyong" fmt.Println(person.Name) // "chenmingyong" } 复制代码
- 通过声明一个结构体变量,隐式创建一个结构体变量,这个结构体变量各个字段值默认为零值,也就是对应类型的默认值。
- 结构体属性的值,可以通过
变量.字段名
的方式获取,同时也可以通过此方式对字段值进行修改。
- 2、使用复合字面值进行显式初始化结构体对象
import "fmt" // Person 定义一个人的结构体 type Person struct { // 姓名 Name string // 年龄 Age int // 性别 Sex string // 身份证号 idNumber string } func main() { person := Person{ "chenmingyong", 18, "男", "xxxxxxxxx", } fmt.Println(person) // {chenmingyong 18 男 xxxxxxxxx} } 复制代码
- 上述代码根据字段的顺序进行赋值,实际上,Go 语言并不推荐我们用这种方式进行赋值,因为可能会带来一些问题,例如结构体的某个中间的字段被删除或者字段的顺序改变了,那么我们需要维护对应代码,调整赋值的顺序。
- 上述创建结构体的方式有弊端,因此我们需要换一种方法,如下第三种方法。
- 3、 使用
field:value
形式的复合字面值进行显式初始化结构体对象
import "fmt" // Person 定义一个人的结构体 type Person struct { // 姓名 Name string // 年龄 Age int // 性别 Sex string // 身份证号 idNumber string } func main() { person := Person{ Sex: "男", Age: 18, Name: "chenmingyong", idNumber: "xxxxxxxxx", } fmt.Println(person) // {chenmingyong 18 男 xxxxxxxxx} } 复制代码
- 通过以上的方式,我们就不被字段的顺序所约束了。
- 4、通过
new(T)
函数创建结构体指针
// Person 定义一个人的结构体 type Person struct { // 姓名 Name string // 年龄 Age int // 性别 Sex string // 身份证号 idNumber string } func main() { person := new(Person) (*person).Name = "chenmignyong" fmt.Println((*person).Name) // chenmignyong // 简化赋值,底层自动转换成 (*person).Age = 18 person.Age = 18 fmt.Println(person.Age) // 18 } 复制代码
- 前面提到过,访问指针所指向变量的值,需要使用
*
操作符,但在结构体这里有点特殊,如果不加*
的话,底层会将person.Age = 18
转成(*person).Age = 18
。
小结
本文对指针和结构体进行了介绍,也指出使用指针和结构体时需要注意的一些地方。因为本文是基于了解的层面去写的文章,一些深入的知识并没有提到,然后也没有提到结构体的方法,是因为打算留到后面和函数一起去介绍。