goang Receiver & interface

diegodu · · 1704 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

package main

import (
    "fmt"
)

type Pointer struct {
    x string
}

func (this *Pointer) PrintX() {
    fmt.Printf("X:%s\n", this.x)
}

func (this *Pointer) SetX(str string) {
    this.x = str 
}

type Value struct {
    x string
}

func (this Value) PrintX() {
    fmt.Printf("X:%s\n", this.x)
}

func (this Value) SetX(str string) {
    this.x = str 
}

func main() {
    p1 := Pointer{x: "Pointer"}
    p1.PrintX()
    p1.SetX("Pointer1")
    p1.PrintX()

    p2 := &Pointer{}
    p2.PrintX()
    p2.SetX("Pointer2")
    p2.PrintX()

    p3 := Value{x: "Value"}
    p3.PrintX()
    p3.SetX("Value1")
    p3.PrintX()

    p4 := &Value{x: "value"}
    p4.PrintX()
    p4.SetX("Value2")
    p4.PrintX()
}

运行结果:

X:Pointer
X:Pointer1
X:
X:Pointer2
X:Value
X:Value
X:value
X:value

说明:1 无论是T*作为receiver还是T类型作为接受者,都可以用实例、或者实例的指针调用函数,但是T*作为接受者将改变receiver的内容,而T类型改变的是副本,原始对象不会改变。

 

type Type struct {
}

type PType struct {
}

type Inter interface {
    post()
}

// 接收者非指针
func (t Type) post() {
}

// 接收者是指针
func (t *PType) post() {
}

func test() {
    var it Inter
    //var it *Inter //接口不能定义为指针
    pty := &Type{}
    it = ty   // 将变量赋值给接口,OK
    it.post() // 接口调用方法,OK, receiver 是T

    ty := Type{}
    it = pty  // 把指针变量赋值给接口,OK
    it.post() // 接口调用方法,OK, receiver 是T

    pty2 := &PType{}
    it = pty2 // 把指针变量赋值给接口,OK
    it.post() // 接口调用方法,OK, receiver 是*T

    //ty2 := PType{}
    //it = ty2  // 将变量赋值给接口,error
    //it.post() // 接口调用方法,error, receiver 必须是T,而不能是T
}

详细代码:http://play.golang.org/p/KG8-Qb7gqM

 

为什么编译器不考虑我们的值是实现该接口的类型?接口的调用规则是建立在这些方法的接受者和接口如何被调用的基础上。下面的是语言规范里定义的规则,这些规则用来说明是否我们一个类型的值或者指针实现了该接口:

 

  • 类型 *T 的可调用方法集包含接受者为 *T 或 T 的所有方法集

 

这条规则说的是如果我们用来调用特定接口方法的接口变量是一个指针类型,那么方法的接受者可以是值类型也可以是指针类型。显然我们的例子不符合该规则,因为我们传入 SendNotification 函数的接口变量是一个值类型。

 

  • 类型 T 的可调用方法集包含接受者为 T 的所有方法

 

这条规则说的是如果我们用来调用特定接口方法的接口变量是一个值类型,那么方法的接受者必须也是值类型该方法才可以被调用。显然我们的例子也不符合这条规则,因为我们 Notify 方法的接受者是一个指针类型。

 

语言规范里只有这两条规则,我通过这两条规则得出了符合我们例子的规则:

 

  • 类型 T 的可调用方法集不包含接受者为 *T 的方法

 

 

 

 

Interface也是引用类型。

 

 

 

 

 


有疑问加站长微信联系(非本文作者)

本文来自:博客园

感谢作者:diegodu

查看原文:goang Receiver & interface

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

1704 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传