继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

Go语言之方法详解

慕粉0415482673
关注TA
已关注
手记 297
粉丝 64
获赞 367

方法是与对象实例绑定的特殊函数。

用于维护和展示对象自身的状态。对象是内敛的。普通函数则专注与算法流程,通过接受参数来完成特定的逻辑运算,并返回最终结果,方法是有关联状态的,函数通常是没有的。

方法和函数定义语法区别在于前者实例接受参数,编译器以此确定方法所属的类型。在一些语言中尽管没有定义,但是函数使用了隐式的传递this实例参数。

可以为当前包,以及除接口和指针以外的任何类型定义方法。方法同样不支持重载,receiver参数名没有限制。不推荐使用this和self。方法可以看做特殊的函数,那么receiver的类型自然可以是基础类型或指针。这会关系到调用时对象实例是否被复制。

不可以使用多级指针调用方法。

指针类型的receiver必须是合法指针(包括nil),或能获取实例地址。


如何选择方法的接收器类型:

使用T:

1.不需要修改状态的小对象或者是固定值。

2.引用类型、字符串、函数等指针包装对象。

使用*T:

1.需要修改实例状态。

2.大对象使用*T,以减少复制成本。

3.如果包含Mutex等同步字段,用*T,避免因为复制造成锁操作无效。

4.其他无法确定全部使用*T。


匿名字段:

方法也会有同名遮蔽问题。但是利用这种特性,可以实现类似的覆盖操作。

方法集:

类型有一个与之相关的方法集,这决定它是否实现某个接口。

类型T方法集合包含所有receiver T方法。

类型*T方法集合包含receiver T + *T方法。

匿名嵌入S,T方法集包含所有receiver S方法。

匿名嵌入*S,T方法集包含所有的receiver S + receiver *S方法。

匿名嵌入S或者*S,*T方法集包含所有receiver S + *S。

表达式:

方法可以分为expression和value两种方法状态。

(1)方法表达式:

通过类型引用的方法表达式会被还原成为普通函数样式,接收器是第一个参数,调用时必须显式传参。至于类型,可以是T或者是*T,只要目标方法存在于该类型方法集中即可。

(2)方法值:

基于实例或者是指针引用的方法值,参数签名不会改变,依旧按照正常方式调用。但是当方法值被赋值给变量或者是作为参数传递时,会立即计算并复制该方法执行所需要的接收器对象,与其绑定,以便在稍后执行时,能隐式传递接收器对象。

编译器会为方法值生成一个包装函数,实现间接调用。至于接收器复制。和闭包的实现方法基本相同,打包成funval,经由DX寄存器传送。当然,如果目标方法的接收器是指针类型,那么被复制的仅仅是指针。只要是接收器参数类型正确,使用nil同样可以执行。

package main


import "fmt"


type N int


func main() {

var number N = 100

result := number.toString()

fmt.Println(result)//d

}


func (number N)toString() string{

return fmt.Sprintf("%s",string(number))

}

运行结果:

d


package main


import "fmt"


type N int


func main() {

var a N = 25

a.value()

a.pointer()

fmt.Printf("a: %p, %v", &a, a)

}

func (n N) value() {

n++

fmt.Printf("v: %p, %v\n", &n, n)

}

func (n *N) pointer() {

(*n)++

fmt.Printf("p: %p, %v\n", n, *n)

}

/*

运行结果:

v: 0xc04204c088, 26

p: 0xc04204c080, 26

a: 0xc04204c080, 2

*/


打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP