【Go语言入门系列】(七)如何使用Go的方法?
【Go语言入门系列】前面的文章:
本文介绍Go语言的方法的使用。
1. 声明
如果你用过面向对象的语言,比如Java,那你肯定对类、对象、成员变量、方法等很熟悉。
简单地来说,类是对一类事物的抽象,成员变量是该事物的属性,方法是该事物具有的行为,对象则是该事物所对应的具体个体。
比如说,狗(类),名字(属性),叫(方法),哮天犬(对象)。
但是Go语言中并没有类,自然也没有面向对象中的成员变量和成员方法。但是Go语言中有类似的概念——结构体,结构体中的字段可以看做类中成员属性。
Go中也有类似于面向对象中方法的概念,也叫方法(method),这种方法其实是一种特殊的函数(function)——带有接收者(receiver)的函数。
方法的声明方式如下:
func (接受者) funcName(参数们) (返回值们)
可以看出方法的声明方式和函数的声明方式差不多,但是多了一个接收者,该接收者是一个结构体类型。下面是一个实例:
package main
import "fmt"
type dog struct {
name string
}
func (d dog) say() {//方法
fmt.Println(d.name + " 汪汪汪。。。方法")
}
func main() {
d := dog{"哮天犬"}
d.watchDoor()
}
运行:
哮天犬 汪汪汪。。。方法
say()是一个方法,d是接收者,是一个结构体类型参数,方法里可以访问接收者的字段:
fmt.Println(d.name + " 汪汪汪。。。方法")
通过.可以调用方法:
d.say()
2. 方法和函数
方法method是具有接收者receiver的特殊函数function。下面的例子展示了method和function之间的区别。
package main
import "fmt"
type dog struct {
name string
}
func (d dog) say() {
fmt.Println(d.name + " 汪汪汪。。。方法")
}
func say(d dog) {
fmt.Println(d.name + " 汪汪汪。。。函数")
}
func main() {
d := dog{"哮天犬"}
d.watchDoor()
watchDoor(d)
}
运行:
哮天犬 汪汪汪。。。方法
哮天犬 汪汪汪。。。函数
你可能会问,在这个例子中,既然方法和函数的运行结果一样,那使用方法岂不是多此一举,为何不继续使用函数?
换一个场景:现在有狗、猫、兔子等动物,他们都会叫,只是叫声不同:
package main
import "fmt"
type dog struct {
name string
}
type cat struct {
name string
}
type rabbit struct {
name string
}
func dogSay(d dog) {
fmt.Println(d.name + " 汪汪汪。。。函数")
}
func catSay(c cat) {
fmt.Println(c.name + " 喵喵喵。。。函数")
}
func rabbitSay(r rabbit) {
fmt.Println(r.name + " 吱吱吱。。。函数")
}
func main() {
d := dog{"哮天犬"}
c := cat{"加菲猫"}
r := rabbit{"玉兔"}
dogSay(d)
catSay(c)
rabbitSay(r)
}
运行:
哮天犬 汪汪汪。。。函数
加菲猫 喵喵喵。。。函数
玉兔 吱吱吱。。。函数
上面的三个函数有什么不妥之处呢?
首先,这三个函数都是用来表示叫这一行为,一般来说函数名都会叫say(),但因为不同的动物,函数名不能相同,为了做区别而做出了改变。
其次,叫这个行为应该属于动物,二者在概念上不能分开。比如,说话这个行为是每个人都具有的,但是说话并不能离开人而独自存在。
此时,方法method的优点就体现了出来:
package main
import "fmt"
type dog struct {
name string
}
type cat struct {
name string
}
type rabbit struct {
name string
}
func (d dog) say() {
fmt.Println(d.name + " 汪汪汪。。。方法")
}
func (c cat) say() {
fmt.Println(c.name + " 喵喵喵。。。方法")
}
func (r rabbit) say() {
fmt.Println(r.name + " 吱吱吱。。。方法")
}
func main() {
d := dog{"哮天犬"}
c := cat{"加菲猫"}
r := rabbit{"玉兔"}
d.say() //调用
c.say()
r.say()
}
运行:
哮天犬 汪汪汪。。。方法
加菲猫 喵喵喵。。。方法
玉兔 吱吱吱。。。方法
三个方法的方法名都一样,每个方法都有一个接受者receiver,这个receiver使方法在概念上属于结构体,就像结构体的字段一样,但是没有写在结构体内。
从这三个方法中可以看出:只要方法的接收者不同,即使方法名相同,方法也不相同。
3. 指针和接收者
接收者可以使用指针,和函数的参数使用指针一样(参考Go语言入门系列(六)之再探函数),接收者使用指针传的是引用,不使用指针传的是值拷贝。看下面一个例子:
package main
import "fmt"
type dog struct {
name string
}
func (d *dog) rename(name string) {
d.name = name
fmt.Println("方法内:" + d.name)
}
func (d dog) rename1(name string) {
d.name = name
fmt.Println("方法内:" + d.name)
}
rename和rename1都是改变名字的方法,一个传引用,一个传值。只有rename能真正改变名字。
func main() {
d := dog{"哮天犬"}
d.rename("小黑黑")
fmt.Println(d.name)
}
运行:
方法内:小黑黑
小黑黑
rename把“哮天犬”改为了“小黑黑”。
func main() {
d := dog{"哮天犬"}
d.rename1("小红红")
fmt.Println(d.name)
}
运行:
方法内:小红红
哮天犬
rename1只在方法内改变了名字,并没有真正改变“哮天犬”。因为rename1接收的是d的一个拷贝。
方法的指针接收者可以进行重定向,什么意思呢?下面用四段代码来说明。
如果函数的参数是一个指针参数,那么该函数就必须接收一个指针才行,如果是值则报错:
package main
import "fmt"
func double(x *int) {
*x = *x * 2
}
func main() {
i := 2
double(&i) //编译正确
double(i) //报错
fmt.Println(i)
}
而如果方法的接收者是一个指针,那么该方法被调用时,接收者既可以是指针,又可以是值:
package main
import "fmt"
func (d *dog) rename(name string) {
d.name = name
fmt.Println("方法内:" + d.name)
}
func main() {
d := dog{"哮天犬"}
d.rename("小黑黑") //接收者是值,编译正确
//(&d).rename("小黑黑") //接收者是指针,编译正确
fmt.Println(d.name)
}
对于指针接收者来说,d.rename("小黑黑")被解释为(&d).rename("小黑黑"),如此一来,我们就不需要在意调用方法的接收者是否为指针类型,因为Go会进行“重定向”。
同理,反过来也可以。
如果函数的参数是值,而不是指针,那么该函数必须接受值,否则会报错:
package main
import "fmt"
func double(x int) {
x = x * 2
}
func main() {
i := 2
p := &i
double(*p) //参数是值,编译正确
//double(p) //参数是指针,报错
fmt.Println(i)
}
而如果方法的接收者是一个值,那么该方法被调用时,接收者既可以是值,又可以是指针:
package main
import "fmt"
func (d dog) rename1(name string) {
d.name = name
fmt.Println("方法内:" + d.name)
}
func main() {
d := dog{"哮天犬"}
p := &d
p.rename1("小红红") //接收者是指针,编译正确
//(*p).rename1("小红红") //接收者是值,编译正确
fmt.Println(d.name)
}
对于值接收者来说,p.rename1("小红红")被解释为(*p).rename1("小红红"),如此一来,我们就不需要在意调用方法的接收者是否为值,因为Go会进行“重定向”。
作者简介
我是行小观,我会在公众号『行人观学』中持续更新Java、Go、数据结构和算法、计算机基础等相关文章。
本文章属于系列文章「Go语言入门系列」,本系列从Go语言基础开始介绍,适合从零开始的初学者。
欢迎关注,我们一起踏上行程。
如有错误,还请指正。
【Go语言入门系列】(七)如何使用Go的方法?的更多相关文章
- 【Go语言入门系列】(八)Go语言是不是面向对象语言?
[Go语言入门系列]前面的文章: [Go语言入门系列](五)指针和结构体的使用 [Go语言入门系列](六)再探函数 [Go语言入门系列](七)如何使用Go的方法? 1. Go是面向对象的语言吗? 在[ ...
- 【Go语言入门系列】(九)写这些就是为了搞懂怎么用接口
[Go语言入门系列]前面的文章: [Go语言入门系列](六)再探函数 [Go语言入门系列](七)如何使用Go的方法? [Go语言入门系列](八)Go语言是不是面向对象语言? 1. 引入例子 如果你使用 ...
- Go语言入门系列(四)之map的使用
本系列前面的文章: Go语言入门系列(一)之Go的安装和使用 Go语言入门系列(二)之基础语法总结 Go语言入门系列(三)之数组和切片 1. 声明 map是一种映射,可以将键(key)映射到值(val ...
- Go语言入门系列(五)之指针和结构体的使用
Go语言入门系列前面的文章: Go语言入门系列(二)之基础语法总结 Go语言入门系列(三)之数组和切片 Go语言入门系列(四)之map的使用 1. 指针 如果你使用过C或C++,那你肯定对指针这个概念 ...
- Go语言入门系列(六)之再探函数
Go语言入门系列前面的文章: Go语言入门系列(三)之数组和切片 Go语言入门系列(四)之map的使用 Go语言入门系列(五)之指针和结构体的使用 在Go语言入门系列(二)之基础语法总结这篇文章中已经 ...
- 【Go语言入门系列】Go语言工作目录介绍及命令工具的使用
[Go语言入门系列]前面的文章: [保姆级教程]手把手教你进行Go语言环境安装及相关VSCode配置 [Go语言入门系列](八)Go语言是不是面向对象语言? [Go语言入门系列](九)写这些就是为了搞 ...
- Maven入门系列(二)--设置中央仓库的方法
原文地址:http://www.codeweblog.com/maven入门系列-二-设置中央仓库的方法/ Maven仓库放在我的文档里好吗?当然不好,重装一次电脑,意味着一切jar都要重新下载和发布 ...
- Go语言入门系列2 基本语法
get download and install packages and dependencies install = compile and install packages and depend ...
- Go语言入门系列1:安装,How to Write Go Code
https://golang.org/doc/code.html src contains Go source files, pkg contains package objects, and bin ...
随机推荐
- CF804D Expected diameter of a tree 树的直径 根号分治
LINK:Expected diameter of a tree 1e5 带根号log 竟然能跑过! 容易想到每次连接两个联通快 快速求出直径 其实是 \(max(D1,D2,f_x+f_y+1)\) ...
- CF R 635 div1 C Kaavi and Magic Spell 区间dp
LINK:Kaavi and Magic Spell 一打CF才知道自己原来这么菜 这题完全没想到. 可以发现 如果dp f[i][j]表示前i个字符匹配T的前j个字符的方案数 此时转移变得异常麻烦 ...
- bzoj 2780 [Spoj]8093 Sevenk Love Oimaster
LINK:Sevenk Love Oimaster 询问一个模式串在多少个文本串中出现过. 考虑广义SAM 统计这种数量问题一般有三种做法. 一种 暴力bitset 这道题可能可以过? 一种 暴力跳p ...
- json-lib无法下载
maven无法下载json-lib 配置一下这个 <classifier>jdk15</classifier> 因为远程提供了两个
- Springboot+Mybatis+Clickhouse+jsp 搭建单体应用项目(三)(添加增删改查)
一.添加增加接口 @ApiResponses(value = { @ApiResponse(code = 200, message = "接口返回成功状态"), @ApiRespo ...
- c++中包含string成员的结构体拷贝导致的double free问题
最近调试代码遇到一个的问题,提示double free,但是找了好久也没有找到释放两次的地方,后来调试发现,是由于使用了一个包含string成员的结构体,这个结构体使用memcpy拷贝导致的问题: 代 ...
- Java高级篇反射和注解
反射是什么? 反射的作用?能带来什么好处? 反射的使用? 注解的使用? 注解和反射配合实战...
- 解读闭包,这次从ECMAScript词法环境,执行上下文说起
对于x年经验的前端仔来说,项目也做了好些个了,各个场景也接触过一些.但是假设真的要跟面试官敞开来撕原理,还是有点慌的.看到很多大神都在手撕各种框架原理还是有点羡慕他们的技术实力,羡慕不如行动,先踏踏实 ...
- Android SharedPreferences存储详解
什么是SharedPreferences存储 一种轻量级的数据保存方式 类似于我们常用的ini文件,用来保存应用程序的一些属性设置.较简单的参数设置. 保存现场:保存用户所作的修改或者自定义参数设定, ...
- HTML基础-01
HTML:超文本标记语言,是一种使用结构化Web网页(标准制定者:W3C)及其内容的标记语言. 发展过程:XHTML5,HTML5,XHTML1.0,HTML4.01,HTML3.2 HTML5特性: ...