注意: 此文章只是我的个人笔记,如有谬误,错误, 请一定指出!


package main

import "fmt"

//IA test for methodset.
type IA interface {
    Value()
    Pointer()
}

//A is a  test type.
type A int

//Value receiver.
func (a A) Value() {

    fmt.Printf("Value:%p,  %d\n", &a, a)
}

//Pointer receiver.
func (a *A) Pointer() {

    fmt.Printf("Pointer:%p,  %d\n", a, *a)
}
func main() {

    //type A method set: (a A) Value()
    var a A = 1
    fmt.Printf("Origin: %p, %d\n", &a, a)
    fmt.Println("Value receiver------------------------------")
    a.Value()
    a.Pointer() //如: (&a).Value(),  a value converted to a Pointer  by go compiler, so value a can call type *A`s  pointer receiver method`

    //type *A method set: func (a A) Value(), func (a *A) Pointer()
    fmt.Println("Pointer receiver---------------------------")
    p := &a
    p.Value()
    p.Pointer()

    fmt.Println("interface call method by type *A-----------------------")
    var ia IA
    ia = &a
    ia.Value()
    ia.Pointer()

    fmt.Println("interface call method by type A-----------------------")
    //对于接口而言, 对于赋值给他的类型,会持有其副本, 即会复制, 且其持有的值不可取地址,会有更严格的类型检查和限制, 不允许做如:A to *A的转换 ,
    //所以当以类型A调用func (a *A) Pointer()时是不允许的, 因为类型A的方法集中不包括func (a *A) Pointer(), 也可以说, 类型A的方法集(method set)是类型*A的一个子集。
    //总结: 当以非接口类型调用pointer receiver method和 value receiver method时 go编译器会帮助做自动转换,如: &a 或 *a ; 所以淡化value receiver和pointer receiver方法集的边界。
    //但是接口不允许做这些自动转换, 一就是一, 严格划清方法集分界,这是golang接口的设计哲学,不是技术问题,
    //a interface must check type A or type *A , do not allow convert it between of them, so type A can not call type *A`s Pointer receiver method`
    ///ia = a //compliation error,coz type not match:  A does not implement IA (Pointer method has pointer receiver)
    //如果改成: ia = &a 就可以了, 因为:类型*A的方法集包括这个方法, 所以与接口匹配.
    //ia.Value()
    //ia.Pointer()

}

//program output:
//Origin: 0xc04200e290, 1 //注意: 与此地址不同的情况 , 皆产生了复制, 这就是value receiver与pointer receiver的区别,关注点.
//Value receiver------------------------------
//Value:0xc04200e2a8,  1
//Pointer:0xc04200e290,  1
//Pointer receiver---------------------------
//Value:0xc04200e2f0,  1
//Pointer:0xc04200e290,  1
//interface call method by type *A-----------------------
//Value:0xc04200e308,  1
//Pointer:0xc04200e290,  1

//interface call method by type A-----------------------




注意: 此文章只是我个人笔记, 如有错漏,请一定指正, 共同学习, 我的邮箱: htyu_0203_39@sina.com