非指针类型的指针方法

rvi*_*nca 3 methods pointers go

根据对这个问题的回应

有关接收器的指针与值的规则是可以在指针和值上调用值方法,但只能在指针上调用指针方法

但实际上我可以对非指针值执行指针方法:

package main

import "fmt"

type car struct {
    wheels int
}

func (c *car) fourWheels() {
    c.wheels = 4
}

func main() {

    var c = car{}
    fmt.Println("Wheels:", c.wheels)
    c.fourWheels()
    // Here i can execute pointer method on non pointer value
    fmt.Println("Wheels:", c.wheels)
}
Run Code Online (Sandbox Code Playgroud)

那么,这里有什么问题?这是一个新功能吗?或者对问题的回答是错误的?

icz*_*cza 8

您正在调用指针值上的"指针方法".在表达式中:

c.fourWheels()
Run Code Online (Sandbox Code Playgroud)

c是类型car(非指针); 由于该car.fourWheels()方法具有指针接收器,并且因为接收器值是非指针并且是可寻址的,所以它是以下的简写:

(&c).fourWheels()
Run Code Online (Sandbox Code Playgroud)

这是在规范:电话:

如果x可寻址的并且&x方法集包含m,x.m()则是简写(&x).m().

该声明:

有关接收器的指针与值的规则是可以在指针和值上调用值方法,但只能在指针上调用指针方法

像这样解释:

  • 如果你有一个值方法,你总是可以调用它:如果你有一个值,它就可以成为接收者; 如果你有一个指针,你总是可以取消引用它以获得准备成为接收者的值.

  • 如果你有一个指针的方法,你可能并不总是能够调用它,如果你只有一个值,因为有几个表达式(其结果)是不可寻址,因此,你将无法获得一个指向它,将被用作接收器; 这样的例子是函数返回值和映射索引表达式.有关详细信息和示例,请参阅如何在Go中存储对操作结果的引用?; 以及如何从函数调用中获取返回值的指针?(当然,你总是可以将它分配给一个局部变量并获取它的地址,但这是一个副本,指针方法只能修改这个副本而不是原始副本.)