go语言学习-结构体

结构体

go语言中的结构体,是一种复合类型,有一组属性构成,这些属性被称为字段。结构体也是值类型,可以使用new来创建。

定义:

type name struct {
    field1 type1
    field2 type2
    ...
}

我们可以看到每一个字段都由一个名字和一个类型构成,不过实际上,如果我们如果不需要使用某个字段时,可以使用”_”来代替它的名字

并且结构体字段可以是任意类型,函数,接口,甚至是结构体本身都可以

使用结构体

定义一个Person结构体

type Person struct {
    name string
    age  int
}

使用

// 字面量形式初始化
var p1 = Person{
    name: "Tom",
    age: 18,
}

var p2 = Person{"Cat", 20}

fmt.Println(p1)          //{Tom 18}                                                                                     
fmt.Println(p1.name)     // Tom
fmt.Println(p2)          // {Cat 20}

p1.age = 20      // 设置p的age字段的值
fmt.Println(p1)     // {Tom 20}
还可以使用new函数来给一个结构体分配内存,并返回指向已分配内存的指针

var p3 *Person   // 声明一个Person类型的指针
p3 = new(Person)   // 分配内存

// 上面两句相当于
p3 := new(Person)

p3.name = "Cat"
p3.age = 10

fmt.Println(p3)       // &{Cat 10}
fmt.Println(p3.name)  // 

我们可以直接使用结构体指针通过”.”来访问结构体的字段,就像直接使用结构体实例一样, go会自动进行转换

还有一种叫做混合字面量的语法来声明,如下,这其实只是一种简写方式,底层还是调用new方法

var p4 = &Person{"Dog", 10}  // 同样返回的是Person类型的指针

fmt.Println(p4)             // &{Dog 10}
fmt.Println(p4.name)        // Dog

匿名字段

go语言的结构体还支持匿名字段,也就是说一个只有类型而没有字段名(连”_”都没有)的字段,被匿名嵌入的也可以是任何类型,此时类型名就是字段的名字,也就是我们可以直接使用类型名为字段名来访问匿名字段.

另外如果匿名字段是另一个结构体,这就叫做内嵌结构体,这个特性可以模拟类似继承的行为。

type Person struct {
    name string
    age  int
}

type Student struct {
    Person
    int
}

// 定义一个Student类型的变量
var s = Student{Person{"gdb", 10}, 10}

// 可以使用如下的方法访问内部结构体中的字段
fmt.Println(s.Person.name)

// 也可以这样访问,go将自动使用Person的name属性,不过如果在Student中也定义了name字段,这里就不能使用了
fmt.Println(s.name)

// 访问int类型的匿名字段,此时类型就是字段的名字
fmt.Println(s.int)

注意:这样如果两个字段有相同的名字时,外部的名字会覆盖内部的;如果同一级别出现相同名字的字段,会出错,需要注意;并且不能同时嵌⼊某⼀类型和其指针类型,因为它们名字相同。

标签

结构体中的字段除了可以有名称和类型以外,还可以有标签。它是一个附属于字段的字符串,可以是文档或其他的重要标记。后面说反射时再说。

方法

之前学习的面向对象语言,比如说Java, Python中,有类的概念,每个类都可以有自己的成员变量,成员方法,它们都是定义在类中的

go语言中的结构体就类似与面向对象语言的类,而结构体的字段就相当于类中的成员变量,结构体也可以有方法,但是不是直接定义在结构体中的,go语言中有一个接收者的概念,我们可以将函数作用在一个接收者,此时这个函数就被称为方法

接收者是某种类型的变量,不仅仅可以是结构体,几乎任何类型都可以是结构体,比如: int,bool, string或数组的别名类型,甚至可以是函数类型,不过不能是接口类型

定义方法的示例:

type Person struct {
    name string
    age int
}

// 使用Person类型的实例做接收者,这就是一个Person类型方法,方法名前面括号中的就是接收者
func (this Person) getName() string {
    return this.name
}

// Peron类型的指针对象也可以作为接收者
func (this *Person) setName(name string) {
    this.name = name
}

tom := Person{"Tom", 20}
fmt.Println(tom)  // {Tom 20}
fmt.Println(tom.getName())  // Tom

tom.changeName("Bob")
fmt.Println(tom)  // {Bob 20}

这里有一点需要注意:类型和绑定它的方法必须在同一个包中(不一定要在同一个文件中)

这里使用类型直接作为接收着 和 类型的指针作为接收者的区别,就相当于普通函数中,值类型的参数和引用类型参数的区别;即在方法中对类型的实例的操作,不会影响外部的实例的值,而使用类型指针的实例作为引用参数,在方法内部修改会影响外部的实例

匿名字段

我们也可以使用结构体内部的匿名字段,作为方法的接收者,此时这个结构体,仍然可以调用这个方法,此时编译器会负责查找

type Person struct {
    name string
    age int
}

type Student struct {
    Person
    score int
}

func (p *Person) show() {
    fmt.Println("My name is " + p.name + ", I'm " + strconv.Itoa(p.age) + " years old")
}


tom := Person{"Tom", 20}
// 调用匿名字段作为接收器的方法
tom.show()   // My name is Tom, I'm 20 years old

在此基础上,我们还可以在结构体上,实现与匿名字段同名的方法,就像面向对象中的重写类似,编译器会先查找结构体实例作为接收器的方法。

方法集

根据定义结构体以及方法的不同,方法集也有所不同,了解他们,对理解接口有帮助

type T struct {
    name string
    age int
}

type G struct {
    T
    action string
}

type S struct {
    *T
    sel string
}
  • 类型 T 的方法集包含所有接收者为 T 的方法
  • 类型*T的方法集包含所有接收者为 T的方法(因为go会自动解引用)和所有接收者为 *T 的方法
  • 类型G包含匿名字段 T, 则G的方法集,仅仅包含T类型的方法集
  • 类型S包含匿名字段 *T,则S的方法集,包含T和*T类型的方法集
posted @ 2018-03-25 17:05  gxyz  阅读(445)  评论(0编辑  收藏  举报