go 成长路上的坑(1)

一、先来看一段代码

package main

import "fmt"

type X struct{}

func (x *X) test(){
	println("h1",x)
}
func main(){
	a := X{} 
	a.test()

	(&X{}).test()

	(X{}).test()
}

猜猜他的结果

二、揭晓答案

package main

import "fmt"

type X struct{}

func (x *X) test(){
	println("h1",x)
}
func main(){
	a := X{} 
	a.test()   // 正确

	(&X{}).test()  // 正确 

	(X{}).test()  // 报错 cannot call pointer method on X literal
}

三、为什么会是这样的

  • 声明赋值后调用指针方法
a :=x
a.test()  //正确
   指针方法可以调用的条件:
       receiver 必须是合法的指针(包括nil) 或者 能够获取实例的地址 
    a 是一个可以寻址的变量 ,所以可以调用test() 指针方法

    When the value is addressable, the language takes care of the common case of invoking a pointer method on a value by inserting the address operator automatically.
    翻译:
    当值是可被寻址的,go语言会处理通常的情况:
        在一个值上面调用它的指针方法,编译器会自动插入一个&取地址操作符
  • (&X{}).test() 正确
   指针方法可以调用的条件:
       receiver 必须是合法的指针(包括nil) 或者 能够获取实例的地址 

    (&X{})  是一个合法的指针
  • (X{}).test() 报错
    变量名 = 右值
    X{}  就是右值
    右值 是不可寻址的(unaddressable) ,所以会报错

四、疑问

(X{}).test()    不可寻址 报错了
为什么
(&X{}).test()   能取到地址了?
posted @ 2019-05-28 12:45  xiaobaiskill  阅读(288)  评论(0编辑  收藏  举报