go中的方法以及结构体
绑定方法
在 Go 语言中,我们无法在结构体内定义方法,那如何给一个结构体定义方法呢,答案是可以使用组合函数的方式来定义结构体方法。它和普通函数的定义方式有些不一样,比如下面这个方法
func (person Profile) FmtProfile() { fmt.Printf("名字:%s\n", person.name) fmt.Printf("年龄:%d\n", person.age) fmt.Printf("性别:%s\n", person.gender) }
其中FmtProfile
是方法名,而(person Profile)
:表示将 FmtProfile 方法与 Profile 的实例绑定。我们把 Profile 称为方法的接收者,而 person 表示实例本身,它相当于 Python 中的 self,在方法内可以使用 person.属性名
的方法来访问实例属性。
方法的参数传递方式
当你想要在方法内改变实例的属性的时候,必须使用指针做为方法的接收者。
package main import "fmt" // 声明一个 Profile 的结构体 type Profile struct { name string age int gender string mother *Profile // 指针 father *Profile // 指针 } // 重点在于这个星号: * func (person *Profile) increase_age() { person.age += 1 } func main() { myself := Profile{name: "小明", age: 24, gender: "male"} fmt.Printf("当前年龄:%d\n", myself.age) myself.increase_age() fmt.Printf("当前年龄:%d", myself.age) }
输出结果 如下,可以看到在方法内部对 age 的修改已经生效。你可以尝试去掉 *
,使用值做为方法接收者,看看age是否会发生改变(答案是:不会改变)
- 当前年龄:24 当前年龄:25
至此,我们知道了两种定义方法的方式:
-
以值做为方法接收者
-
以指针做为方法接收者
那我们如何进行选择呢?以下几种情况,应当直接使用指针做为方法的接收者。
-
你需要在方法内部改变结构体内容的时候
-
出于性能的问题,当结构体过大的时候
有些情况下,以值或指针做为接收者都可以,但是考虑到代码一致性,建议都使用指针做为接收者。
不管你使用哪种方法定义方法,指针实例对象、值实例对象都可以直接调用,而没有什么约束。这一点Go语言做得非常好。
内部方法与外部方法
在 Go 语言中,函数名的首字母大小写非常重要,它被来实现控制对方法的访问权限。
-
当方法的首字母为大写时,这个方法对于所有包都是Public,其他包可以随意调用
-
当方法的首字母为小写时,这个方法是Private,其他包是无法访问的。
方法定义:
Golang 方法总是绑定对象实例,并隐式将实例作为第一实参 (receiver)。
• 只能为当前包内命名类型定义方法。
• 参数 receiver 可任意命名。如方法中未曾使用 ,可省略参数名。
• 参数 receiver 类型可以是 T 或 *T。基类型 T 不能是接口或指针。
• 不支持方法重载,receiver 只是参数签名的组成部分。
• 可用实例 value 或 pointer 调用全部方法,编译器自动转换。
一个方法就是一个包含了接受者的函数,接受者可以是命名类型或者结构体类型的一个值或者是一个指针。
所有给定类型的方法属于该类型的方法集。
普通函数与方法的区别
1.对于普通函数,接收者为值类型时,不能将指针类型的数据直接传递,反之亦然。
2.对于方法(如struct的方法),接收者为值类型时,可以直接用指针类型的变量调用方法,反过来同样也可以。
package main
//普通函数与方法的区别(在接收者分别为值类型和指针类型的时候)
import (
"fmt"
)
//1.普通函数
//接收值类型参数的函数
func valueIntTest(a int) int {
return a + 10
}
//接收指针类型参数的函数
func pointerIntTest(a *int) int {
return *a + 10
}
func structTestValue() {
a := 2
fmt.Println("valueIntTest:", valueIntTest(a))
//函数的参数为值类型,则不能直接将指针作为参数传递
//fmt.Println("valueIntTest:", valueIntTest(&a))
//compile error: cannot use &a (type *int) as type int in function argument
b := 5
fmt.Println("pointerIntTest:", pointerIntTest(&b))
//同样,当函数的参数为指针类型时,也不能直接将值类型作为参数传递
//fmt.Println("pointerIntTest:", pointerIntTest(&b))
//compile error:cannot use b (type int) as type *int in function argument
}
//2.方法
type PersonD struct {
id int
name string
}
//接收者为值类型
func (p PersonD) valueShowName() {
fmt.Println(p.name)
}
//接收者为指针类型
func (p *PersonD) pointShowName() {
fmt.Println(p.name)
}
func structTestFunc() {
//值类型调用方法
personValue := PersonD{101, "hello world"}
personValue.valueShowName()
personValue.pointShowName()
//指针类型调用方法
personPointer := &PersonD{102, "hello golang"}
personPointer.valueShowName()
personPointer.pointShowName()
//与普通函数不同,接收者为指针类型和值类型的方法,指针类型和值类型的变量均可相互调用
}
func main() {
structTestValue()
structTestFunc()
}
接收者是指针类型的方法,很可能在方法中会对接收者的属性进行更改操作,从而影响接收者;而对于接收者是值类型的方法,在方法中不会对接收者本身产生影响。
最后,只要记住下面这点就可以了:
package main
import "fmt"
type Person struct {
age int
}
func (p Person) howOld() int {
return p.age
}
func (p *Person) growUp() {
p.age += 1
}
func (p Person) growUpValue() {
p.age += 1
}
func main() {
// qcrao 是值类型
qcrao := Person{age: 18}
// 值类型 调用接收者也是值类型的方法
fmt.Println(qcrao.howOld())
// 值类型 调用接收者是指针类型的方法
qcrao.growUp()
fmt.Println(qcrao.howOld())
qcrao.growUpValue()
fmt.Println(qcrao.howOld())
// ----------------------
// stefno 是指针类型
stefno := &Person{age: 100}
// 指针类型 调用接收者是值类型的方法
fmt.Println(stefno.howOld())
// 指针类型 调用接收者也是指针类型的方法
stefno.growUp()
fmt.Println(stefno.howOld())
stefno.growUpValue()
fmt.Println(stefno.howOld())
}
18
19
19
100
101
101
实际上,当类型和方法的接收者类型不同时,其实是编译器在背后做了一些工作,用一个表格来呈现:
- | 值接收者 | 指针接收者 |
---|---|---|
值类型调用者 | 方法会使用调用者的一个副本,类似于“传值” | 使用值的引用来调用方法,上例中,qcrao.growUp() 实际上是 (&qcrao).growUp() |
指针类型调用者 | 指针被解引用为值,上例中,stefno.howOld() 实际上是 (*stefno).howOld() |
实际上也是“传值”,方法里的操作会影响到调用者,类似于指针传参,拷贝了一份指针 |
- 实现了接收者是值类型的方法,相当于自动实现了接收者是指针类型的方法;
- 而实现了接收者是指针类型的方法,不会自动生成对应接收者是值类型的方法。
如果实现了接收者是值类型的方法,会隐含地也实现了接收者是指针类型的方法。
package main
import "fmt"
type coder interface {
code()
debug()
}
type Gopher struct {
language string
}
func (p Gopher) code() {
fmt.Printf("I am coding %s language\n", p.language)
}
func (p *Gopher) debug() {
fmt.Printf("I am debuging %s language\n", p.language)
}
func main() {
var c coder = &Gopher{"Go"}
c.code()
c.debug()
}
运行一下,结果:
|
|
但是如果我们把 main
函数的第一条语句换一下:
|
/go_receiver.go:26:18: cannot use Gopher{…} (value of type Gopher) as coder value in variable declaration: Gopher does not implement coder (method debug has pointer receiver) 也就是 var c coder = Gopher{"Go"} 这句报错了 |
第一次是将 &Gopher
赋给了 coder
;第二次则是将 Gopher
赋给了 coder
。
第二次报错是说,Gopher
没有实现 coder
。很明显了吧,因为 Gopher
类型并没有实现 debug
方法;表面上看, *Gopher
类型也没有实现 code
方法,但是因为 Gopher
类型实现了 code
方法,所以让 *Gopher
类型自动拥有了 code
方法。
当然,上面的说法有一个简单的解释:接收者是指针类型的方法,很可能在方法中会对接收者的属性进行更改操作,从而影响接收者;而对于接收者是值类型的方法,在方法中不会对接收者本身产生影响。
两者分别在何时使用 #
如果方法的接收者是值类型,无论调用者是对象还是对象指针,修改的都是对象的副本,不影响调用者;如果方法的接收者是指针类型,则调用者修改的是指针指向的对象本身。
使用指针作为方法的接收者的理由:
- 方法能够修改接收者指向的值。
- 避免在每次调用方法时复制该值,在值的类型为大型结构体时,这样做会更加高效
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 25岁的心里话
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· 零经验选手,Compose 一天开发一款小游戏!
· 一起来玩mcp_server_sqlite,让AI帮你做增删改查!!
2020-05-10 icmp port unreachable
2020-05-10 sock skbuf 结构:
2020-05-10 连接建立定时器