struct
1. Go 语言中,也和 C 或者其他语言一样,我们可以声明新的类型,作为其它类型的属性或字段的容器。
2. 例如,我们可以创建一个自定义类型 person 代表一个人的实体。这个实体拥有属性:姓名和年龄。
3. 这样的类型我们称之 struct。
type person struct {
name string
age int
}
看到了吗?声明一个 struct 如此简单,上面的类型包含有两个字段
1. 一个 string 类型的字段 name,用来保存用户名称这个属性
2. 一个 int 类型的字段 age, 用来保存用户年龄这个属性
如何使用 struct 呢?
type person struct {
name string
age int
}
func main() {
per := person{"rod", 18}
fmt.Println(per)
per.name = "aod"
per.age = 20
fmt.Println(per)
p := new(person)
p.name = "bod"
p.age = 12
fmt.Println(p)
fmt.Println(*p)
}
{rod 18}
{aod 20}
&{bod 12}
{bod 12}
几种声明使用方式:
1. 按照顺序提供初始化值 P := person{"Tom", 25}
2. 通过 field:value 的方式初始化,这样可以任意顺序P := person{age:24, name:"Tom"}
3. 当然也可以通过 new 函数分配一个指针,此处 P 的类型为 *person P := new(person)
struct 的匿名字段
1. 我们上面介绍了如何定义一个 struct,定义的时候是字段名与其类型一一对应,实际上 Go 支持只提供类型,而不写字段名的方式,也就是匿名字段,也称为嵌入字段。
2. 当匿名字段是一个 struct 的时候,那么这个 struct 所拥有的全部字段都被隐式地引入了当前定义的这个 struct。
type person struct {
name string
age int
}
type Student struct {
person // 匿名字段,那么默认 Student 就包含了 person 的所有字段
speciality string
}
func main() {
student := Student{person{"rod", 20}, "学生"}
fmt.Println(student)
fmt.Println(student.name)
}
{{rod 20} 学生}
rod
如图
1. 我们看到 Student 访问属性 age 和 name 的时候,就像访问自己所有用的字段一样,对,匿名字段就是这样,能够实现字段的继承。
2. 是不是很酷啊?还有比这个更酷的呢,那就是 student 还能访问 Human 这个字段作为字段名。
请看下面的代码,是不是更酷了。
student.person = person{"aod", 11}
fmt.Println(student)
struct 不仅仅能够将 struct 作为匿名字段,自定义类型、内置类型都可以作为匿名字段
1. 这里有一个问题:
2. 如果 human 里面有一个字段叫做 phone,而 student 也有一个字段叫做 phone,那么该怎么办呢?
3. Go 里面很简单的解决了这个问题,最外层的优先访问,也就是当你通过 访问的时候,是访问 student 里面的字段,而不是 human 里面的字段。
4. 这样就允许我们去重载通过匿名字段继承的一些字段,当然如果我们想访问重载后对应匿名类型里面的字段,可以通过匿名字段名来访问。
type Human struct {
name string
age int
phone string // Human 类型拥有的字段
}
type Employee struct {
Human // 匿名字段 Human
speciality string
phone string // 雇员的 phone 字段
}
func main() {
Bob := Employee{Human{"Bob", 34, "777-444-XXXX"}, "Designer", "333-222"}
fmt.Println("Bob's work phone is:", Bob.phone)
// 如果我们要访问 Human 的 phone 字段
fmt.Println("Bob's personal phone is:", Bob.Human.phone)
}
Bob's work phone is: 333-222
Bob's personal phone is: 777-444-XXXX