+-
一、结构体
结构体是一系列属性的集合(类似于 Python 中的类)
1、结构体的定义与使用
// 定义 type Person struct { Name string Age int Sex string } func main() { // 使用 var per Person per.Name="XiaoYang" fmt.Println(per) }
2、定义并赋初值
type Person struct { Name string Age int Sex string } func main() { var per1 Person = Person{Name: "XiaoYang"} // 按关键字传参,可以少传 var per2 Person = Person{"Bob", 20, "男"} // 按位置传参,全传 fmt.Println(per1) // 输出:{XiaoYang 0 } fmt.Println(per2) // 输出:{Bob 20 男} }
3、匿名结构体(只使用一次,没有名字)
// 定义个匿名结构体并实例化之后赋值给了 hobby 变量 hobby := struct { HobbyId int HobbyName string }{HobbyId: 1, HobbyName: "篮球"} fmt.Println(hobby) // 输出:{1 篮球} fmt.Println(hobby.HobbyName) // 输出:篮球
4、结构体的零值
定义好的结构体没有被初始化时,该结构体的字段将默认赋值为零值
也就是我属性的零值,所以他是值类型,参数传递,copy 传递,在函数中修改不会影响原来的
type Person struct { Name string Age int Sex string } func main() { var per Person=Person{"Bob", 20, "男"} fmt.Println(per) // 输出:{Bob 20 男} test(per) // 输出:{Bob 20 男} fmt.Println(per) // 输出:{Bob 20 男} } func test(per Person) { per.Age=20 fmt.Println(per) }
5、结构体的指针
// & 放在变量前,表示取该变量的地址 // * 放在类型前,表示指向该类型的指针(变量定义,指定类型时才会用到) *[3]int 和 [3]*int // * 放在变量前,表示解引用(取出指针指向的具体的值) type Person struct { Name string Age int Sex string } func main() { var per1 *Person fmt.Println(per1) // 输出:<nil> 表示指针类型 // 定义并初始化 var per2 *Person = &Person{} fmt.Println(per2) // 输出:&{ 0 } // 把per2的名字改成XiaoYang (*per2).Name = "XiaoYang" // 也支持直接使用 per2.Name = "Bob" fmt.Println(per2) // 输出:&{Bob 0 } }
6、匿名字段(字段没有名字,只有类型)
可用于【变量提升 / 提升字段】类似于面向对象的继承
// 定义一个结构体,匿名字段类型就是字段名字,所有类型不能重复 type Person struct { string int Sex string } func main() { per := Person{"XiaoYang", 20, "男"} // 字段匿名,类型就是字段名 fmt.Println(per) // 输出:{XiaoYang 20 男} fmt.Println(per.string) // 输出:XiaoYang }
7、嵌套结构体(结构体中套结构体)
type Person struct { Name string Age int Sex string Hobby Hobby // Person中嵌套Hobby结构体字段 } type Hobby struct { HobbyId int HobbyName string } func main() { per := Person{Name:"XiaoYang", Age: 20, Sex: "男", Hobby: Hobby{1,"篮球"}} fmt.Println(per) // 输出:{XiaoYang 20 男 {1 篮球}} fmt.Println(per.Name) // 输出:XiaoYang fmt.Println(per.Hobby.HobbyName) // 输出:篮球 }
8、字段提升
结构体中有匿名的结构体类型字段,则该匿名结构体里的字段就称为提升字段,可以从外部直接访问
type Person struct { Name string Age int Sex string Hobby // Person中嵌套Hobby匿名结构体字段 } type Hobby struct { HobbyId int HobbyName string } func main() { per := Person{Name:"XiaoYang", Age: 20, Sex: "男", Hobby: Hobby{1,"篮球"}} // 打印爱好的名字(Hobby是一个匿名字段,会字段提升) fmt.Println(per.HobbyName) // 输出:篮球 // per.hobby 类似于面向对象中的super() fmt.Println(per.Hobby.HobbyName) // 输出:篮球 } // ------------------------------------------------------------------------------------ // 当子类和父类中的字段名一样时,就像面向对象的继承,子类继承父类(结构体嵌套,匿名字段),子类可以直接调用父类中的属性或方法 type Person struct { Name string Age int Sex string Hobby // Person中嵌套Hobby匿名字段 } type Hobby struct { HobbyId int Name string } func main() { per := Person{Name:"XiaoYang", Age: 20, Sex: "男", Hobby: Hobby{1,"篮球"}} fmt.Println(per.Name) // 输出:XiaoYang ——>优先使用自己的 fmt.Println(per.Hobby.Name) // 输出:篮球 ——>指定打印Hobby的名字 }
9、结构体相等性
结构体是值类型。
如果它的每一个字段都是可比较的,则该结构体也是可以比较的。
如果两个结构体变量的对应字段相等,则两个变量也是相等的。
如果结构体包含不可比较的字段,则结构体变量也不可比较。
type Person struct { Name string Age int Sex string // 包含不可比较的字段 Test []int ——>这是引用类型不可比较 } func main() { // 值类型可以直接==比较,引用类型只能跟nil用==比较 per1 := Person{Name: "XiaoYang"} per2 := Person{Name: "XiaoYang"} per3 := Person{Name: "XiaoYang", Age: 20} fmt.Println(per1 == per2) // 输出:ture ——>包含不可比较类型都会直接报错 fmt.Println(per1 == per3) // 输出:false }
二、方法
方法就是一个特殊函数,在函数的基础上加了一些东西
在
func这个关键字和方法名中间加入一个特殊的接收器类型,接收器可以是结构体类型,也可以是非结构体类型
1、方法的定义和使用
type Person struct { Name string Age int Sex string } // 定义一个方法:给Person结构体绑定一个方法,oneself(名字随便取)类似于Python类中的self func (oneself Person) printName() { // 在方法内使用oneself fmt.Println(oneself.Name) } func main() { // 使用,对象调用方法 per := Person{} per.Name = "XiaoYang" // 绑定给对象的方法 per.printName() // 输出:XiaoYang }
2、有了函数为啥还需要方法?
方法功能都能实现,但是呢?它就能指定给某个对象了 。
type Person struct { Name string Age int Sex string } // 方法 func (oneself Person) printName() { fmt.Println(oneself.Name) } // 函数 func printName(oneself Person) { fmt.Println(oneself.Name) } func main() { per := Person{Name: "XiaoYang"} per.printName() // 方法的特殊之处,可以自动传递值 printName(per) // 函数需要手动传递值 }
3、指针接收器与值接收器
type Person struct { Name string Age int Sex string } // 值接收器修改名字 func (oneself Person) changeName(name string) { oneself.Name = name } // 指针接收器修改年龄 func (oneself *Person) changeAge(age int) { oneself.Age = age } func main() { per := Person{Name: "XiaoYang", Age: 20} fmt.Println(per) // 输出:{XiaoYang 20 } per.changeName("Bob")// 由于这个是值接收器,它是copy一份传递过去所以修改的是copy的不会改掉原来的 per.changeAge(18) // 指针接收器,它传递的是指针 fmt.Println(per) // 输出:{XiaoYang 18 } } /* 什么时候用指针接收器,什么时候使用值接收器: -想改原来的,就用指针 -不想改原来的,就用值 */
5、匿名字段的方法(方法提升)
type Person struct { Name string Age int Sex string Hobby // 匿名字段 } type Hobby struct { Id int Name string } // 打印Person的名字 func (oneself Person) printName() { fmt.Println(oneself.Name) } // 打印Hobby的名字 func (oneself Hobby) printHobbyName() { fmt.Println(oneself.Name) } // 打印Hobby的名字 func (oneself Hobby) printName() { fmt.Println(oneself.Name) } func main() { per := Person{Name: "XiaoYang", Hobby: Hobby{1, "篮球"}} per.printName() // 输出:XiaoYang per.printHobbyName() // 输出:篮球 // 如果方法重名了,优先使用结构体自己的 per.printName() // 输出:XiaoYang per.Hobby.printName() // 输出:篮球 }
6、在方法中使用值接收器 与 在函数中使用值参数
type Person struct { Name string Age int Sex string } // 在方法中使用值接收器 func (oneself Person) printName() { fmt.Println(oneself.Name) } // 在函数中使用值参数 func printName(oneself Person) { fmt.Println(oneself.Name) } func main() { per1 := &Person{Name: "XiaoYang"} // per1是个指针 per2 := Person{Name: "Bob"} printName(*per1) // 输出:XiaoYang per1.printName() // 输出:XiaoYang ———> 值收器:可以用值来调,也可以用指针来调 per2.printName() // 输出:Bob }
7、在方法中使用指针接收器 与 在函数中使用指针参数
type Person struct { Name string Age int Sex string } // 在方法中使用指针接收器 func (oneself *Person) printName() { fmt.Println(oneself.Name) } func (oneself *Person)changeName(name string) { oneself.Name=name } // 在函数中使用指针参数 func printName(oneself *Person) { fmt.Println(oneself.Name) } func main() { per1 := Person{Name: "XiaoYang"} per2 := &Person{Name: "Bob"} // per1是个指针 per1.printName() // 值可以调用 printName(&per1) per2.printName() // 指针可以调用 printName(per2) per1.changeName("Alen") // 可以修改 fmt.Println(per1) per2.changeName("YS") // 可以修改 fmt.Println(per2) } /* 总结: -不管是值类型接收器还是指针类型接收器,都可以用值来调用,或者指针来调用。 -不管是值还是指针来调用,只要是值类型接收器,改的就是新的,只要是指针类型接收器,改的是原来的。 */
8、非结构体上绑定方法
不允许在基础数据类型上绑定方法(如:int、string ... )
但是自己定义的类型可以绑定方法
type Myint int func (i *Myint) add() { (*i)++ } func main() { var a Myint = 10 fmt.Println(a) // 输出:10 a.add() a.add() a.add() fmt.Println(a) // 输出:13 }