【注意】最后更新于 March 1, 2023,文中内容可能已过时,请谨慎使用。
Go没有沿袭传统面向对象编程中的诸多概念,也没有提供类(class)
,但是它提供了结构体(struct)
,方法(method)
可以在结构体上添加。与类相似,结构体提供了捆绑数据和方法的行为。
1.介绍
1.1 概念
Go语言同时有函数和方法,方法的本质是函数,但是方法和函数又有所不同。
1.2 方法和函数的区别
- 函数
(function)
是一段具有独立功能的代码,可以被反复多次调用,从而实现代码复用。
- 方法
(method)
是一个类的行为功能,只有该类的对象才能调用。
- 方法有接受者,而函数无接受者。
- 函数不可以重名,而方法可以重名。只要接受者不同,方法名就可以相同。
2. 使用
2.1 基本语法
1
2
3
|
func (接收器变量 接收器类型) 方法名(参数列表) (返回参数) {
函数体
}
|
- 接收器变量:接收器变量在命名时,官方建议使用接收器类型的第一个小写字母,而不是
self、this
之类的命名。例如: Socket类型的接收器变量应该命名为s,Connector类型的接收器变量应该命名为c等。
- 接收器类型:接收器类型和参数类似,可以是指针类型和非指针类型。
- 方法名、参数列表、返回参数:格式与函数定义一致。
2.2 使用示例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
package main
import "fmt"
// 定义一个结构体
type Student struct {
name string
age int
}
// 定义一个方法(接收器为Student的指针)
func (s *Student)updateName(newName string) {
s.name = newName
}
// 定义一个方法(接收器为Student)
func (s Student) updateAge(newAge int) {
s.age = newAge
fmt.Printf("修改结构体s的age -> %v \n",s)
}
func main() {
// 初始化结构体
s := Student{"张三",20}
fmt.Printf("结构体初始化s -> %v \n",s)
// 通过方法修改名称
s.updateName("张三新名")
fmt.Printf("调用updateName后 -> %v \n",s)
// 通过方法修改年龄
s.updateAge(22)
fmt.Printf("调用updateAge后 -> %v \n",s)
}
/** 输出:
结构体初始化s -> {张三 20}
调用updateName后 -> {张三新名 20}
修改结构体s的age -> {张三新名 22}
调用updateAge后 -> {张三新名 20}
*/
|
通过上述示例可以看出:若方法的接受者不是指针,实际只是获取了一个拷贝,而不能真正改变接受者中原来的数据。
3.方法继承
方法是可以继承的,如果匿名字段实现了一个方法,那么包含这个匿名字段的struct也能调用该匿名字段中的方法。
3.1 使用示例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
package main
import "fmt"
// 定义一个人类结构体
type People struct {
name, position string
age int
}
// 定义一个学生结构体
type Student struct {
People
}
type Teacher struct {
People
}
// 定义一个方法
func (p People) say() {
fmt.Printf("我叫 %s %d岁 从事: %s \n", p.name,p.age,p.position)
}
func main() {
student := Student{People{"张三","学生",15}}
teacher := Teacher{People{"李杨","老师",35}}
// 调用方法(继承父类)
student.say()
teacher.say()
}
/** 输出
我叫 张三 15岁 从事: 学生
我叫 李杨 35岁 从事: 老师
*/
|
4 .方法重写
在Go语言中,方法重写是指一个包含了匿名字段的struct
也实现了该匿名字段实现的方法(即子类也实现了父类的方法)
4.1 使用示例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
package main
import "fmt"
// 定义一个人类结构体
type People struct {
name, position string
age int
}
// 定义一个学生结构体
type Student struct {
People
}
// 定义一个老师结构体
type Teacher struct {
People
}
// 定义一个方法
func (p People) say() {
fmt.Printf("我叫 %s %d岁 从事: %s \n", p.name, p.age, p.position)
}
// 学生(子类)重写People(父类)的say方法
func (s Student) say() {
fmt.Printf("我是一名学生,名字叫: %s 今年: %d岁 \n", s.name, s.age)
}
func main() {
student := Student{People{"张三", "学生", 15}}
teacher := Teacher{People{"李杨", "老师", 35}}
// 调用方法(重写父类方法)
student.say()
// 调用方法(继承父类)
teacher.say()
}
/** 输出
我是一名学生,名字叫: 张三 今年: 15岁
我叫 李杨 35岁 从事: 老师
*/
|
当结构体存在继承关系时,方法调用按照就近原则。