希望解决的问题如下

假设有两个方法,一个方法的接收者是指针类型,一个方法的接收者是值类型,那么:

  • 对于值类型的变量和指针类型的变量,这两个方法有什么区别?
  • 如果这两个方法是为了实现一个接口,那么这两个方法都可以调用吗?
  • 如果方法是嵌入到其他结构体中的,那么上面两种情况又是怎样的?
值类型的变量和指针类型的变量

先声明一个结构体:

type T struct {
    Name string
}

func (t T) M1() {
    t.Name = "name1"
}

func (t *T) M2() {
    t.Name = "name2"
}

M1() 的接收者是值类型 T, M2() 的接收者是值类型 *T , 两个方法内都是改变Name值。

TM1()M2()
    t1 := T{"t1"}

    fmt.Println("M1调用前:", t1.Name)
    t1.M1()
    fmt.Println("M1调用后:", t1.Name)

    fmt.Println("M2调用前:", t1.Name)
    t1.M2()
    fmt.Println("M2调用后:", t1.Name)

输出结果为:

M1调用前: t1
M1调用后: t1
M2调用前: t1
M2调用后: name2

下面猜测一下go会怎么处理。

func M1(t T)func
 M2(t *T)
t1.M1()M1(t1)
t1.M2()M2(t1)M2(&t1)

T 类型的变量这两个方法都是拥有的。

*TM1()M2()
    t2 := &T{"t2"}

    fmt.Println("M1调用前:", t2.Name)
    t2.M1()
    fmt.Println("M1调用后:", t2.Name)

    fmt.Println("M2调用前:", t2.Name)
    t2.M2()
    fmt.Println("M2调用后:", t2.Name)

输出结果为:

M1调用前: t2
M1调用后: t2
M2调用前: t2
M2调用后: name2
t2.M1()M1(t2)
t2.M2()M2(t2)

*T 类型的变量也是拥有这两个方法的。

传给接口会怎样?

先声明一个接口

type Intf interface {
    M1()
    M2()
}

使用:

    var t1 T = T{"t1"}
    t1.M1()
    t1.M2()

    var t2 Intf = t1
    t2.M1()
    t2.M2()

报错:

./main.go:9: cannot use t1 (type T) as type Intf in assignment:
    T does not implement Intf (M2 method has pointer receiver)
var t2 Intf = t1

t1 是有 M2() 方法的,但是为什么传给 t2 时传不过去呢?

t1 是值类型,赋值给 t2 时是复制值而不是指针,假设 t1 可以赋值给 t2, t2.M2() 修改 Name 的值时也是修改的拷贝的变量,无法影响到 t1,那把 t1 赋值给 t2 还有什么意义呢?所以这种赋值是不被允许的。

var t2 Intf = t1var
 t2 Intf = &t1t2.M2()
func f(t Intf)
嵌套类型

声明一个类型 S,将 T 嵌入进去

type S struct {
    T
}

使用下面的例子测试一下:

    t1 := T{"t1"}
    s := S{t1}

    fmt.Println("M1调用前:", s.Name)
    s.M1()
    fmt.Println("M1调用后:", s.Name)

    fmt.Println("M2调用前:", s.Name)
    s.M2()
    fmt.Println("M2调用后:", s.Name)

    fmt.Println(t1.Name)

输出:

M1调用前: t1
M1调用后: t1
M2调用前: t1
M2调用后: name2
t1

将 T 嵌入 S, 那么 T 拥有的方法和属性 S 也是拥有的,但是接收者却不是 S 而是 T。

s.M1()M1(t1)M1(s)
S{t1}

假如我们将 s 赋值给 Intf 接口会怎么样呢?

    var intf Intf = s
    intf.M1()
    intf.M2()

报错:

cannot use s (type S) as type Intf in assignment:
    S does not implement Intf (M2 method has pointer receiver)

还是 M2() 的问题,因为 s 此时还是值类型。

var intf Intf = &sintf.M2()s.Namet1.Name

下面嵌入 *T 试试:

type S struct {
    *T
}

使用时这样:

    t1 := T{"t1"}
    s := S{&t1}

    fmt.Println("M1调用前:", s.Name)
    s.M1()
    fmt.Println("M1调用后:", s.Name)

    fmt.Println("M2调用前:", s.Name)
    s.M2()
    fmt.Println("M2调用后:", s.Name)

    fmt.Println(t1.Name)

惟一的区别是最后 t1 的值变了,因为我们复制的是指针。

接着赋值给接口试试:

    var intf Intf = s
    intf.M1()
    intf.M2()
    fmt.Println(s.Name)

编译没有报错。这里我们传递给 intf 的是值类型而不是指针,为什么可以通过呢?

拷贝 s 的时候里面的 T 是指针类型,所以调用 M2() 的时候传递进去的是一个指针。

var intf Intf = &s