初心是记录和总结,自己学习Go语言的历程。如果能帮助到你,这是我的荣幸。
属于一个类型的方法
这里出现一个新的概念:方法。在go语言中,方法和函数还是有区别的:
- 函数是传入特定值或直接调用后输出结果或直接执行某种操作
- 而方法是作用于特定类型变量的函数
举个清晰的例子,在我的上一篇文章中,写了一个例子:描述的是给person
这个自定义类型创建一个setter
方法,使得它可以通过setter
对成员进行赋值。当时我们的操作是这样的:
func main() {
type person struct {
name string //描述姓名
age int //描述年龄
setNameAndAge func(p *person, name string, age int)
}
var people person //定义person类型的变量为people
people.setNameAndAge = func(p *person, name_ string, age_ int) {
p.name = name_
p.age = age_
fmt.Printf("%p\n", p) //0xc0000543c0
}
people.setNameAndAge(&people, "张三", 15)
fmt.Printf("%p\n", &people) //0xc0000543c0
fmt.Println(people) //{张三 15 0xa4fc80}
}
这样的操作,需要我们在结构体中定义好变量之后,还要为变量初始化:赋值一个方法。比较麻烦不说,当我们输出该变量的时候还会发现方法也被输出了。那前面我们说:方法是作用于特定类型变量的函数,我们来看看方法是怎么解决这一类事情的:结构体的方法。
方法的语法
func (接收变量形参名称 接收类型) 方法名(参数列表)(返回值列表){
}
可以看出它和函数的语法很像,唯一区别的地方在于(接收变量形参名称 接收类型)
,这里就是表明了什么类型可以接收(或调用)该方法。也就是说当我们这样写的时候,就可以通过变量名.方法名
调用了方法了!
func (p person) 方法名(参数列表)(返回值列表){
}
var people person
people.方法名
使用方法绑定给struct类型
介绍完毕后,我们来实现通过方法实现:给person
这个自定义类型创建一个setter
方法。
package main
import "fmt"
type person struct {
name string //描述姓名
age int //描述年龄
}
func main() {
// 定义
var people person
people.setter("程云来", 18)
fmt.Println(people) //{程云来 18}
}
func (people *person) setter(name string, age int) {
people.name = name
people.age = age
}
(people *person)
是将操作的地址进行传递,保证了操作的是同一块内存地址的内容,而不使用指针的方式,一般都是值传递。而变量一般是有作用域范围的,每个学过编程语言的都会有概念,值传递的作用域范围一般在方法中都是方法内局部使用的,如果不将值作为范围值传递回去用参数接收,它在方法执行完毕后就会被垃圾回收
。
既然我们上面都说了指针和作用域范围以及值和地址传递的概念,下面我们通过函数以及方法了解一下这些概念的使用。
指针传递和值传递
我们通过一个例子来感受一下。定义int
变量num
,分别调用usePoint
函数以及notUsePoint
两个函数,这两个函数的区别一个是指针形参
另一个是类型形参
。
函数调用结束后,我们会发现:
- 使用指针的函数改变了原
num
的值 - 使用类型形参的函数没有改变
num
的值
package main
import "fmt"
func main() {
num := 0
notUsePoint(num)
fmt.Println(num) //0
usePoint(&num)
fmt.Println(num) //10
}
func usePoint(num *int) {
*num = 10
}
func notUsePoint(num int) {
num = 10
}
那么类似的,上面的例子改用方法其实是一样的。但是! 方法有个奇妙的用法并且和函数不同的,记住就行:
- 接收者为指针类型和值类型的方法,指针类型和值类型的变量均可相互调用。但是函数不行!
回到我们之前的例子:
func main(){
people.setter("程云来", 18)
fmt.Println(people)
}
func (people *person) setter(name string, age int) {
people.name = name
people.age = age
}
发现一个关键要点没!我的方法接受者
是指针类型,而值类型变量
可以直接调用,但是操作的时候是将值类型变量的地址传递过来了。但是函数形参和传递参数类型不一致的时候就会报错。
可以理解方法中的接受者
是方法定义的规则,按照我的规则,我是操作值还是操作地址。