• Golang 匿名字段


    定义一个struct,定义的时候是字段名与其类型一一对应,实际上Go语言支持只提供类型,而不写字段名的方式,也就是匿名字段,或称为嵌入字段。

    当匿名字段是一个struct的时候,那么这个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{"Mark", 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)
    
    }

    图2.7  Student和Human的方法继承的内存结构

    我们看到Student访问属性age和name的时候,就像访问自己所有用的字段一样,对,匿名字段就是这样,能够实现字段的继承。是不是很酷?还有比这个更酷的呢!那就是student还能访问Human这个字段作为字段名,请看下面的代码。

    mark.Human = Human{"Marcus", 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() {
    
        // 初始化学生Jane
    
        jane := Student{Human:Human{"Jane", 35, 100}, speciality:"Biology"}
    
        // 现在我们来访问相应的字段
    
        fmt.Println("Her name is ", jane.name)
    
        fmt.Println("Her age is ", jane.age)
    
        fmt.Println("Her weight is ", jane.weight)
    
        fmt.Println("Her speciality is ", jane.speciality)
    
        // 我们来修改他的skill技能字段
    
        jane.Skills = []string{"anatomy"}
    
        fmt.Println("Her skills are ", jane.Skills)
    
        fmt.Println("She acquired two new ones ")
    
        jane.Skills = append(jane.Skills, "physics", "golang")
    
        fmt.Println("Her skills now are ", jane.Skills)
    
        // 修改匿名内置类型字段
    
        jane.int = 3
    
        fmt.Println("Her preferred number is", jane.int)
    
    }
    

      

    从上面例子可见,struct不仅能将struct作为匿名字段,自定义类型、内置类型都可以作为匿名字段,而且可以在相应的字段上进行函数操作(如例子中的append)。

    有个问题:如果human里面有一个字段叫做phone,而student也有一个字段叫做phone,那么该怎么办呢?

    Go语言很简单地解决了这个问题,最外层的优先访问,也就是当你通过student.phone访问的时候,是访问student里面的字段,而不是human里面的字段。

    这样就允许我们去重载通过匿名字段继承的一些字段,当然如果我们想访问重载后对应匿名类型里面的字段,可以通过匿名字段名来访问。请看下面的例子。

    package main
    
    import "fmt"
    
    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)
    
    }
    

      

    关于对方法的继承,看一段的代码对比就比较明了了:

    package main
    
    import (
    	"fmt"
    )
    
    type Animal struct {
    	Age int
    }
    
    type Person struct {
    	Animal
    	Sex int
    
    }
    
    func (animal Animal) getName() string {
    	return "animal"
    }
    
    // func (person Person) getName() string {
    // 	return "person"
    // }
    
    func main() {
    	var animal Animal
    	var person Person
    	fmt.Println(animal.getName())
    	fmt.Println(person.getName())
    }
    // 输出
    // animal
    // animal
    

      

    package main
    
    import (
    	"fmt"
    )
    
    type Animal struct {
    	Age int
    }
    
    type Person struct {
    	Animal
    	Sex int
    
    }
    
    func (animal Animal) getName() string {
    	return "animal"
    }
    
    func (person Person) getName() string {
    	return "person"
    }
    
    func main() {
    	var animal Animal
    	var person Person
    	fmt.Println(animal.getName())
    	fmt.Println(person.getName())
    }
    // 输出
    // animal
    // persion
    

      最后,如果匿名字段是指针的话,那么只是占一个指针的大小:

    package main
    import "fmt"
    import "unsafe"
    
    type T struct {
    	a int
    	b float64
    }
    
    type A struct{
    	T
    }
    
    type B struct{
    	*T
    }
    
    func main() {
    	var t T
    	var a A
    	var b B
    	fmt.Println(unsafe.Sizeof(t))
    	fmt.Println(unsafe.Sizeof(a))
    	fmt.Println(unsafe.Sizeof(b))
    
    	var x *T
    	fmt.Println(unsafe.Sizeof(x))
    }
    // 输出:
    // 16
    // 16
    // 8
    // 8
    

      

  • 相关阅读:
    C语言I博客作业07
    C语言I 博客作业03
    我的第一次作业
    C语言I博客作业05
    js闭包在你身边却不知
    浅谈SQL Transaction在请求中断后的行锁表锁
    事件订阅分发模型
    js 继承 对象方法与原型方法
    PHP学习笔记有关php中的变量
    PHP学习笔记不同编码占据不同的字节
  • 原文地址:https://www.cnblogs.com/maji233/p/11059589.html
Copyright © 2020-2023  润新知