浅谈Go语言函数与方法的区别

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了浅谈Go语言函数与方法的区别相关的知识,希望对你有一定的参考价值。

参考技术A     前段时间,我们实验室用go作为后台开发语言开发了一个web项目,由于这是自己第一次使用go语言进行开发,在开发过程中,一味着追求完成任务,在编码的时候没有太注重性能,虽然勉强实现了功能,但是对go语言的理解还是比较浅显的。下面来谈谈自己对go语言中函数与方法的理解。

普通函数:

    go函数可以返回多个值

    值传递: 值传递是指在调用函数时将实际参数复制一份传递到函数中,这样函数中如果对参数进行修改,将不会影响到实际参数

    引用传递: 引用传递是指在调用函数将实际参数的地址传递到函数中,那么在函数中对参数进行的修改,将影响到实际参数。

一般来说go语言函数的 接收者(也就是形参)一般放在函数名后面 ,不能将指针类型的数据直接传递,也就是说函数形参如果是值类型,调用者必须使用值作为实参过来,如果函数形参是指针类型,则函数调用者需使用指针作为实参来调用。

普通方法:

接收者是在func关键字后面,而不是在函数名称后面,接收者可以是自己定义的一个类型,这个类型可以是struct、interface,一个方法就是一个包含了接收者的函数,接收者可以是命名类型或者是结构体类型的一个值或者是一个指针。

下面是一个例子来说明方法和函数的区别(重点)

Go语言之方法详解

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

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

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

*/


以上是关于浅谈Go语言函数与方法的区别的主要内容,如果未能解决你的问题,请参考以下文章

浅谈Go语言中的结构体struct & 接口Interface & 反射

C语言与Go语言有什么区别?

C语言与Go语言有什么区别?

Go语言之方法详解

go语言使用go-sciter创建桌面应用 事件处理,函数与方法定义,go与tiscript之间相互调用

GO语言实战之函数与方法