什么是方法?
方法与函数的区别是,函数不属于任何类型,方法属于特定的类型。
普通类型的方法接受者
Go语言中的 方法(Method) 是一种作用于特定类型变量的函数。这种特定类型变量叫做 接收者(Receiver)。
接收者的概念就类似于其他语言中的this或者 self。
方法的定义格式如下:
func (接收者变量 接收者类型) 方法名(参数列表) (返回参数) {
函数体
}
其中,接收者变量:接收者中的参数变量名在命名时,官方建议使用接收者类型名称首字母的小写,而不是self、this之类的命名。例如,Person类型的接收者变量应该命名为 p,Connector类型的接收者变量应该命名为c等。
接收者类型:接收者类型和参数类似,可以是指针类型和非指针类型。
方法名、参数列表、返回参数:具体格式与函数定义相同。
举个栗子,func ( x DDDD) FOO(var type) ( ret type , err Errot),这里来说一下 x DDDD是什么意思。
在go语言中,没有类的概念但是可以给类型(结构体,自定义类型)定义方法。
所谓方法就是定义了接受者的函数,方法和函数只差了一个,那就是方法在 func
和标识符之间多了一个参数。接受者定义在func关键字和函数名之间:
type Person struct {
name string
age int
}
func (p Person) say() {
fmt.Printf("I'm %s,%d years old\n",p.name,p.age)
}
有了对方法及接受者的简单认识之后,接下来主要谈一下接受者的类型问题。
接收者有两种,一种是值接收者,一种是指针接收者。顾名思义,值接收者,是接收者的类型是一个值,是一个副本,方法内部无法对其真正的接收者做更改;指针接收者,接收者的类型是一个指针,是接收者的引用,对这个引用的修改之间影响真正的接收者。
情况一:接受者是struct
package main
import "fmt"
type Person struct {
name string
age int
}
func (p Person) say() {
fmt.Printf("I'm %s,%d years old\n",p.name,p.age)
}
func (p Person) older(){
p.age = p.age +1
}
func main() {
var p1 Person = Person{"zhansan",16}
p1.older()
p1.say()
//output: I'm zhangsan,16 years old
var p2 *Person = &Person{"lisi",17}
p2.older()
p2.say()
//output: I'm lisi,17 years old
}
对于p1的调用,读者应该不会有什么疑问。
对于p2的调用可能存在这样的疑问,p2明明是个指针,为什么再调用了older方法之后,打印结果还是17 years old?
方法的接受者是Person而调用者是*Person ,其实在p2调用时存在一个转换p2.older() -> *p2.older(); p2.say() -> *p2.say()
*p2是什么想必读者也是明白的(就一个p2指向Person实例)。那么疑问也就自然的解开了,方法执行时的接受者实际上还是一个值而非引用。
情况二:接受者是指针
package main
import "fmt"
type Person struct {
name string
age int
}
func (p *Person) say() {
fmt.Printf("I'm %s,%d years old\n",p.name,p.age)
}
func (p *Person) older(){
p.age = p.age +1
}
func main() {
var p1 Person = Person{"zhansan",16}
p1.older()
p1.say()
//output: I'm zhangsan,17 years old
var p2 *Person = &Person{"lisi",17}
p2.older()
p2.say()
//output: I'm lisi,18 years old
}
p1的调用中也存在一个转换,
p1.older -> *p1.older
p1.say() -> *p1.say()
在举一个清晰的调用例子:
我们参照着来写一下 Go 的方法定义。
首先,我们是先要定义一个类型,比如就是 user
好了,然后我们再定义方法。
type user struct {
name string
email string
}
func (u user) notify() {
fmt.Println("Email is %d", u.email)
}
func (u *user) changeEmail(email string) {
u.email = email
}
我们定义了两个方法,一个是 notify
,它是值接收者方法;还有一个是 changeEmail
,它是指针接收者方法。可以看到,值接收者方法,接收者是一个副本,无法修改;指针接收者是引用,可以修改。
我们再来看一下调用。
daryl := {"daryl", "daryl@oldexample.com"}
daryl.changeEmail("daryl@example.com")
daryl.notify()
看看,是不是很熟悉!对,就像 php
代码一样,有没有!daryl
就是对象,name
和 email
就是属性,notify
和 changeEmail
就是它的方法。只是,不同的是,我们没有将它放到 class
中,而是用另外一种方式让它们结合了,有了关系!
再举一个例子:
type T struct {
Name string
}func (t T) M1() {
t.Name = “name1”
}func (t *T) M2() {
t.Name = “name2”
}
M1() 的接收者是值类型 T, M2() 的接收者是值类型 *T , 两个方法内都是改变Name值。
下面声明一个 T
类型的变量,并调用 M1()
和 M2()
。
t1 := T{“t1”}
fmt.Println(“M1调用前:”, t1.Name)
t1.M1()
fmt.Println(“M1调用后:”, t1.Name)fmt.Println(“M2调用前:”, t1.Name)
t1.M2()
fmt.Println(“M2调用后:”, t1.Name)
M1调用前: t1
M1调用后: t1
M2调用前: t1
M2调用后: name2
先来约定一下:接收者可以看作是函数的第一个参数,即这样的: func M1(t T)
, func M2(t *T)
。 go不是面向对象的语言,所以用那种看起来像面向对象的语法来理解可能有偏差。
当调用 t1.M1()
时相当于 M1(t1)
,实参和行参都是类型 T,可以接受。此时在M1()中的t只是t1的值拷贝,所以M1()的修改影响不到t1。
当调用 t1.M2()
=> M2(t1)
,这是将 T 类型传给了 *T 类型,go可能会取 t1 的地址传进去: M2(&t1)
。所以 M2() 的修改可以影响 t1 。
如需转载请注明: 转载自26点的博客
本文链接地址: Go基础:函数声明之方法接受者(函数名之前括号中的内容)
转载请注明:26点的博客 » Go基础:函数声明之方法接受者(函数名之前括号中的内容)