struct类型
struct
Go语言中,也和C或者其他语言一样,我们可以声明新的类型,作为其他类型的实行或字段的容器,例如,我们可以创建一个自定义类型person代表一个人的实体,这个实体拥有的属性:姓名和年龄.这样的类型我们称之为struct.如下代码所示:
type person struct { name string age int }
上面的类型包含了两个字段
一个string类型的字段name,用来保存用户名称这个属性
一个int类型的字段age,用来保护存用户年龄这个属性
如何使用struct呢?
package main import "fmt" type person struct { name string age int } var P person // P现在就是person类型的变量 func main() { P.name = "陈润" //赋值"陈润"给P的name P.age = 25 // 赋值25给变量P的age属性 fmt.Printf("The person's name is %s", P.name) // 访问P的name属性 }
除了上面这种P的声明使用之外,还有另外几种声明是有方式:
1,按照顺序提供初始化
P := person{"Tom", 25}
2,通过field:value的方式初始化,这样可以任意顺序
P := person{age:24, name: "Tom"}
3,当然也可以通过new函数分配一个指针,此处P的类型为*person
P := new(person)
下面我们看一个完整的struct的例子

package main import ( "fmt" ) type person struct { name string age int } //比较两个人的年龄,返回年龄大的那个人,并且返会年龄差 //struct也是传值的 func Older(p1, p2 person) (person, int) { if p1.age > p2.age { return p1, p1.age - p2.age } return p2, p2.age - p1.age } func main() { var tom person //赋值初始化 tom.name, tom.age = "Tom", 18 //两个字段都写清楚的初始化 bob := person{age:25, name:"Bob"} //按照struct定义顺序初始化值 paul := person{"Paul", 43} tb_Older, tb_diff := Older(tom, bob) tp_Older, tp_diff := Older(tom, paul) bp_Older, bp_diff := Older(bob, paul) fmt.Printf("of %s and %s, %s is older by %d years\n", tom.name, bob.name, tb_Older.name, tb_diff) fmt.Printf("of %s and %s, %s is older by %d years\n", tom.name, paul.name, tp_Older.name, tp_diff) fmt.Printf("of %s and %s, %s is older by %d years\n", bob.name, paul.name, bp_Older.name, bp_diff) }
struct的函数字段
我们上面介绍了如何定义一个struct,定义的时候是字段名与其类型一一对应,实际上Go支持只提供类型,而不写字段名的方式,u二九四以命字段,也称为嵌入字段.
当匿名字段是一个sutrct的时候,那么这个struct所拥有的全部字段都被隐式的引入了当前定义的额这个struct
看这个例子

package main import "fmt" type Human struct { name string age int weight int } type Student struct { Human // 匿名字段,那么默认student就包含了Human的所有字段 speciality string } func main() { // 我们初始化一个学生 mark := Student{Human{"chenrun", 25, 120}, "computer science"} fmt.Println("His name is ", mark.name) fmt.Println("His age is ", mark.age) fmt.Println("His weight is ", mark.weight) fmt.Println("His speciality is ", mark.speciality) //修改对应的备注信息 mark.speciality = "AI" fmt.Println("Mark changed his speciality") fmt.Println("his speciality is ", mark.speciality) //修改他的年龄信息 fmt.Println("mark become old") mark.age = 46 fmt.Println("his age is ", mark.age) //修改他的体重信息 fmt.Println("mark is not an athlet anymore") mark.weight += 60 fmt.Println("his weight is", mark.weight) }
struct组合,student组合了Human struct和string基本类型
我们看到Student访问属性age和name的时候,就像访问自己所有用的字段一样,匿名字段就是这样,能够实现字段的继承,是不是很酷,还有比这个更酷的,那就是student还能访问Human这个字段作为字段名.请看下面这个代码
mark.Human = Human{"chenrun", 55, 220} mark.Human.age -= 1
通过匿名访问和修改字段相当有用,但是补给你进士struct字段哦,所有的内置类型和自定义类型都是可以作为匿名字段的.请看下面代码

package main import "fmt" type Skills []string type Human struct { name string age int weight int } type student struct { Human // 匿名字段,struct Skills // 匿名字段,自定义的类型,string slice int // 内置类型作为匿名字段 speciality string } func main() { // 初始化一个学生"陈润" cr := student{Human: Human{"陈润", 20, 100}, speciality: "Biology"} //现在我们来访问相应的字段 fmt.Println("Her name is ", cr.name) fmt.Println("her age is ", cr.name) fmt.Println("her weight is ", cr.weight) //我们来修改他的skill技能字段 cr.Skills = []string{"anatomy"} fmt.Println("her skills are ", cr.Skills) fmt.Println("she acquired two new ones") cr.Skills = append(cr.Skills, "physics", "golang") fmt.Println("her skills now are ", cr.Skills) //修改匿名内置类型字段 cr.int = 3 fmt.Println("her perferred number is ", cr.int) }
从上面的例子我们看出来struct不仅仅能够将struct作为匿名字段,自定义类型,内置类型都可以作为匿名字段,而且可以在相应的字段上面进行函数操作
这里有一个问题,如果human里面有一个字段叫做phone,而student也有一个字段叫做phone,那么该怎么办呢?
Go里面很简单的解决了这个问题,最外层的优先访问,也就是当你通过student.phone访问的时候,是访问student里面的字段,而不是human里面的字段
这样就允许我们取重载通过匿名字段继承的一些字段,当然如果我们想访问重载后对应匿名类型里面的字段,可以通过匿名字段来访问.请看下面例子

package main import "fmt" type Human struct { name string age int phone string } type Employee struct { Human speciality string phone string // 雇员的phone字段 } func main() { Bob := Employee{Human{"Bob", 34, "17610780919"}, "Designer", "333-444"} fmt.Println("bob is work phone is ", Bob.phone) // 如果我们要访问Human的phone字段 fmt.Println("bob is personal phone is ", Bob.Human.phone) }