go tour --方法值接收器和指针接收器区别

有两个理由使用指针接收器

  1. 第一个是为了让方法可以修改它的接收器指向的值。
  2. 第二种方法是避免在每次方法调用时复制值。例如,如果接收方是一个大的结构体,这可能会更有效率。

通常,给定类型上的所有方法都应该具有值或指针接收器,但不能两者混合。(我们将在接下来的几页中看到原因。)

接口

接口是定义的一组方法签名。

实现接口必须啊实现接口的所有方法。

实现接口必须要值接收器方法。

在底层,一个接口值可以被认为是一个值和具体类型的元组

(value, type)

调用接口值上的方法将在其底层类型上执行同名方法。

如果接口内部的具体值本身是nil,该方法将用nil接收器调用。(但还是会调用方法,因为有类型,可以在方法内通过判断接收器为nil 来避免空指针错误)

package main

import "fmt"

type I interface {
        M()
}

type T struct {
        S string
}

func (t *T) M() {
        if t == nil {
                fmt.Println("<nil>")
                return
        }
        fmt.Println(t.S)
}

func main() {
        var i I

        var t *T
        i = t
        describe(i)
        i.M()

        i = &T{"hello"}
        describe(i)
        i.M()
}

func describe(i I) {
        fmt.Printf("(%v, %T)\n", i, i)
}

注意,持有nil具体值的接口值本身是非nil的。

nil接口值既不包含值也不包含具体类型。

在nil接口上调用方法是一个运行时错误,因为在接口元组中没有类型来指示要调用哪个具体方法。

package main

import "fmt"

type I interface {
        M()
}

func main() {
        var i I
        describe(i)
        i.M()
}

func describe(i I) {
        fmt.Printf("(%v, %T)\n", i, i)
}


// =========================

(<nil>, <nil>)
panic: runtime error: invalid memory address or nil pointer dereference
[signal 0xc0000005 code=0x0 addr=0x0 pc=0x49c106]

goroutine 1 [running]:
main.main()

指定0个方法的接口类型称为空接口

interface{}

空接口可以保存任何类型的值。(每种类型至少实现0个方法。)

空接口由处理未知类型值的代码使用。例如,fmt.Print接受 interface{} 类型的任意数量的参数。

package main

import "fmt"

func main() {
        var i interface{}
        describe(i)

        i = 42
        describe(i)

        i = "hello"
        describe(i)
}

func describe(i interface{}) {
        fmt.Printf("(%v, %T)\n", i, i)
}
// =========================
(<nil>, <nil>)
(42, int)
(hello, string)